記錄--千萬別讓 console.log 上生產!用 Performance 和 Memory 告訴你為什麼

来源:https://www.cnblogs.com/smileZAZ/archive/2023/02/10/17109664.html
-Advertisement-
Play Games

這裡給大家分享我在網上總結出來的一些知識,希望對大家有所幫助 很多前端都喜歡用 console.log 調試,先不談調試效率怎麼樣,首先 console.log 有個致命的問題:會導致記憶體泄漏。 為什麼這麼說呢? 用 Performance 和 Memory 工具分析下就知道了。 我們準備這樣一段代 ...


這裡給大家分享我在網上總結出來的一些知識,希望對大家有所幫助

很多前端都喜歡用 console.log 調試,先不談調試效率怎麼樣,首先 console.log 有個致命的問題:會導致記憶體泄漏。

為什麼這麼說呢?

用 Performance 和 Memory 工具分析下就知道了。

我們準備這樣一段代碼:

一個按鈕,點擊之後創建一個數組,執行一些計算。

很常見的邏輯。

我們最後加了一個 console.log 列印了下這個數組。

起個靜態服務:

瀏覽器訪問:

 

 點擊 performance 下的垃圾回收按鈕,手動觸發一次 GC:

 

 

勾選 Memory,然後開始錄製,點擊 3 次按鈕,再執行一次 GC:

你會發現記憶體是這樣的:

記憶體占用有三次增長,因為我們點擊三次按鈕的時候會創建 3 次大數組。

但是最後我們手動 GC 之後並沒有回落下去,也就是這個大數組沒有被回收。

按理來說,代碼執行完,那用的記憶體就要被釋放,然後再執行別的代碼,結果這段代碼執行完之後大數組依然占據著記憶體,這樣別的代碼再執行的時候可用記憶體就少了。

這就是發生了記憶體泄漏,也就是代碼執行完了不釋放記憶體的流氓行為。

有同學說,只是這麼一點記憶體問題不大呀,反正可用記憶體還很多。

但如果你的代碼要跑很長時間,這段代碼要執行很多次呢?

每次執行都會占據一部分記憶體不釋放,慢慢的記憶體就不夠用了,甚至會導致程式崩潰。

比如當這段代碼執行個 9 次,記憶體占用就增長了 9 個大數組的記憶體:

再多執行幾次呢?

是不是就有崩潰的隱患了。

那為啥說是 console.log 導致的呢?

我們來看看不用 console.log 是什麼樣的:

註釋掉 console.log,重新跑。

你會發現現在的記憶體分配情況是這樣的:

分配了三次記憶體,但是 GC 後又會落下去了。

這才是沒有記憶體泄漏的好代碼。

那為啥 console.log 會導致記憶體泄漏呢?

因為控制台列印的對象,你是不是有可能展開看?那如果這個對象在記憶體中沒有了,是不是就看不到了?

所以有這個引用在,瀏覽器不會把你列印的對象的記憶體釋放掉。

有的同學說,那我不打開控制台,是不是就沒有這個引用了?

答案是否定的:

2023-01-05 18.34.31.gif

我點擊了幾次之後,再打開控制台,依然是可以看到這個對象的,說明沒有被 GC。

也就是說用 console.log 列印對象的代碼一定是有記憶體泄漏的。

當然,也不只是 console.log 會導致記憶體泄漏,還有別的 4 種情況:

  • 定時器用完了沒有清除,那每次執行都會多一個定時器的記憶體占用,這就是記憶體泄漏
  • 元素從 dom 移除了,但是還有一個變數引用著他,這樣的游離的 dom 元素也不會被回收。每執行一次代碼,就會多出游離的 dom 元素的記憶體,這也是記憶體泄漏
  • 閉包引用了某個變數,這個變數不會被回收,如果這樣的閉包比較多,那每次執行都會多出這些被引用變數的記憶體占用。這樣引用大對象的閉包多了之後,也會導致記憶體問題
  • 全局變數,這個本來就不會被 GC,要註意全局變數的使用

總之,全局變數、閉包引用的變數、被移除的 dom 依然被引用、定時器用完了沒清除、console.log 都會發生代碼執行完了,但是還占用著一部分記憶體的流氓行為,也就是記憶體泄漏。

註意,這裡指的是使用完畢後沒有回收,在使用期間的記憶體增長是正常的。

那怎麼排查呢?

performance 工具就可以:

點擊記憶體分配情況的某個點,就會定位到 performance 中的某個任務的代碼,點擊可以在下麵看到詳情:

這樣就定位到了分配記憶體的代碼,分析一下哪裡會有問題即可。

當然,前提還是要執行先 GC,再做一些操作,再 GC 的這個流程。

這是從代碼角度來分析記憶體泄漏,其實還可以從記憶體中對象的角度,這個是通過 Memory 工具:

先 GC,錄製一次記憶體快照,再點擊幾次按鈕,然後 GC,再錄製一次記憶體快照。

流程和用 performance 分析的時候一樣。

拿到兩次記憶體快照也是可以分析出有記憶體泄漏的:

可以看到 GC 後記憶體占用依然增長了。

快照記錄著這個時刻記憶體中所有對象的狀態:

對比兩次快照,就可以找到變化的部分:

比如這時候可以看到最大的記憶體增長是 array 對象:

然後就可以從 array 的角度去思考是什麼導致的記憶體泄漏了。

此外,memory 還有實時分析的工具:

選擇第二個,然後點幾次按鈕:

其實不用手動 GC,JS 引擎會做 GC。

去掉 console.log 再錄製是這樣的:

除了最開始全局變數會分配一些記憶體以外,點擊按鈕之後的記憶體變藍後又變灰了,也就是被 GC 了。

這樣你點多少次按鈕,記憶體占用都沒有增長。

這就是代碼執行完,會回收所有用到的記憶體的好代碼。

而前面的那個是每次代碼執行,都會占用一部分記憶體不釋放的記憶體泄漏代碼。

你還可以看到每一次記憶體分配的對象是啥:

不管是用 Performance 工具還是 Memory 工具,都可以發現 console.log 有記憶體泄漏的問題。所以還是儘量不要用這個來調試了。

那應該用什麼呢?

用 debugger 呀,不管是 vscode debugger 還是 chrome devtools 的都可以:

你可以添加一個 logpoint 來代替 console.log 列印:

代碼執行到這裡就會列印:

而你的代碼里不需要寫 console.log。

此外,很多地方可以用斷點代替列印:

可以看到代碼執行路線和作用域,豈不是更高效?

總結

console.log 會導致記憶體泄漏,也就是代碼執行完了,但還占據著一部分記憶體的流氓行為。

除了 console.log,游離的 dom 被變數引用、全局變數、變數被閉包引用、定時器沒清除也會導致記憶體泄漏。

我們可以用 Performance 工具和 Memory 工具分析記憶體泄漏。

先手動 GC,然後執行一些操作,再 GC,如果記憶體沒有回到執行前,就說明這段代碼有記憶體泄漏,可以再用 Performance 定位到代碼位置分析代碼。

Memory 工具是從記憶體對象的角度分析,可以對兩次快照做 diff,看下是啥對象泄漏了。

也可以實時檢測記憶體占用情況,看看是否存在記憶體泄漏,對象是啥。

console.log 調試效率也不高,可以換成 logpoint,或者打斷點。

千萬不要把 console.log 上生產!不然這樣有記憶體泄漏的代碼,一旦執行時間長了就會有問題。

其實普通項目也還好,不會長期跑,但是類似大屏項目這種長期跑的,一旦有記憶體泄漏,一定會崩潰,只有時間長短的區別。

本文轉載於:

https://juejin.cn/post/7185128318235541563

如果對您有所幫助,歡迎您點個關註,我會定時更新技術文檔,大家一起討論學習,一起進步。

 


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 摘要:本文主要講解DWS函數出參帶出方式。 本文分享自華為雲社區《GaussDB(DWS)功能 -- 函數出參 #【玩轉PB級數倉GaussDB(DWS)】》,作者:譡里個檔 。 DWS的PL/pgSQL函數/存儲過程中有一個特殊的語法PERFORM語法,用於執行語句但是丟棄執行結果的場景,常用於一 ...
  • 視圖的增刪改查 視圖相當於一張只能讀的表,不可以修改。當組成視圖的表發生數據變化的時候,視圖會相對應的進行改變。 存儲過程的練習 創建存儲過程: create [if not exists] procedure 名字 ([in | out | inout] 參數名稱 參數類型) begin # sq ...
  • 2023-02-10 一、集群的定義 1、redis集群實現了對redis的水平擴容,即啟動N個redis節點,將整個資料庫分佈存儲在N個節點中,每個節點存儲總數據的1/N。 2、redis集群通過分區來提供一定程度的可用性:即使集群中有一部分節點失效或者無法進行通訊,集群也可以繼續處理命令請求 二 ...
  • 數據治理是推動大型集團企業轉型升級、提升競爭優勢、實現高質量發展的重要引擎。通過全鏈數據結構化,實現業務對象、業務規則、業務流程數字化,推進全鏈業務深度數字化,夯實數據運營底座。 某大型實業集團創立於1980年,主要業務涵蓋供應鏈運營、城市建設與運營、旅游會展、 醫療健康、新興產業投資等多個領域。位 ...
  • 2023-02-10 一、redis提供了2個不同形式的持久化方式 1、RDB(Redis DataBase) 2、AOF(Append Of File) 二、RDB的定義 RDB是在指定的時間間隔內將記憶體中的數據集快照寫入磁碟,即Snapshot快照,它恢復時是將快照文件直接讀到記憶體里。 三、備份 ...
  • 大家都知道 MySQL 的數據都是保存在磁碟的,那具體是保存在哪個文件呢?MySQL 存儲的行為是由存儲引擎實現的,MySQL 支持多種存儲引擎,不同的存儲引擎保存的文件自然也不同。InnoDB 是我們常用的存儲引擎,也是 MySQL 預設的存儲引擎。本文主要以 InnoDB 存儲引擎展開討論。 ...
  • NineData 是多雲數據管理平臺(https://www.ninedata.cloud/),致力於讓每個人用好數據和雲。作為資料庫領域的技術創新團隊,面對這麼火ChatGPT,我們 NineData 的工程師也針對ChatGPT,做了一些關於資料庫領域的相關測試,測試結果,真的是智商狂飆。 ...
  • vuex是什麼? vuex是管理應用程式狀態,實現組件間通信的。 為什麼使用vuex? 在開發大型應用的項目時,會出現多個視圖組件依賴一個同一個狀態,來自不同視圖的行為需要變更同一個狀態。 在遇到以上問題,就要用到vuex,他能把組件的共用狀態抽取出來,當做一個全局單例模式進行管理,不管在何處改變狀 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...