淺析Echarts圖表渲染導致內(nèi)存泄漏的原因及解決方案
一. 引言
在今年某個可視化大屏項目中,出現(xiàn)了一個問題。項目在運行一段時間后,頁面出現(xiàn)了崩潰,而且是大概運行幾天之后,因為大屏項目是部署到客戶現(xiàn)場大屏,長時間運行不關閉。報錯問題如下圖所示:
由于這個大屏頁面是使用 Echarts 圖表渲染的,整個頁面就一個渲染地圖的功能,沒有多少耗費內(nèi)存的操作。而 Echarts 作為一款強大的數(shù)據(jù)可視化庫,被廣泛應用于各種項目中,因此 Echarts 的穩(wěn)定性也是經(jīng)過廣大開發(fā)者驗證的,不會出現(xiàn)明顯的 bug 問題。
然而,我們也知道,如果頁面隨著圖表數(shù)量的增加和動態(tài)更新的需求,我們必然會考慮 Echarts 圖表渲染導致的內(nèi)存泄漏問題。因此,上面的問題我們按照這個猜想來往下走,認為 Echarts 圖表渲染導致了內(nèi)存泄漏,那么 Echarts 的什么操作會導致內(nèi)存泄漏、甚至到頁面崩潰呢?
本文將深入分析這一問題,并提供解決方案。
二. 問題背景
猜測原因
當我們使用 Echarts 來渲染大量圖表時,會發(fā)現(xiàn)頁面的內(nèi)存占用不斷增加,最終導致頁面卡頓甚至崩潰。這是由于 Echarts 在圖表渲染過程中產(chǎn)生的內(nèi)存泄漏導致的。
而奇怪的是,自己在本地中測試的時候,從未有過崩潰的現(xiàn)象?但是在客戶現(xiàn)場就出現(xiàn)了這種問題。
內(nèi)存占用檢測
為了能夠?qū)崟r監(jiān)測網(wǎng)站運行過程中,我們在 Chrome 瀏覽器打開了內(nèi)存性能分析,便于實時查看內(nèi)存占用情況,發(fā)現(xiàn)運行內(nèi)存占用并不是太大,我們監(jiān)測過一段時間后發(fā)現(xiàn),確實內(nèi)存有時在穩(wěn)定的增加,但是這點內(nèi)存占用還不至于導致系統(tǒng)崩潰。
內(nèi)存泄漏原因分析
嵌套事件綁定:在 Echarts 圖表中,每個圖表都會綁定各種交互事件。如果我們正確地解綁這些事件,則會導致事件監(jiān)聽器無法垃圾回收,從而造成內(nèi)存泄漏。
大量的圖表實例:如果我們在動態(tài)更新圖表的過程中,每次都創(chuàng)建新的圖表實例而不銷毀舊的實例,就會導致內(nèi)存占用不斷增加。
定時器未清理:如果我們在更新圖表的過程中使用了定時器來控制刷新頻率,但未能正確清理這些定時器,就可能導致內(nèi)存泄漏。
三. 第一輪解決方案
因為可視化大屏項目代碼不是我直接寫的,我當時只不過是被臨時抽過去提過幾個建議而已,所以并沒有深入的研讀項目代碼,所以這一輪解決方案可以定義為是在浪費時間,不過也并不是一點經(jīng)驗沒有,一起來看一下吧。
初步解決
為了解決 Echarts 圖表渲染導致的內(nèi)存泄漏問題,我們采取以下措施:
在綁定事件監(jiān)聽器時,要確保正確解綁,可以使用 Echarts 提供的
off
方法來取消事件定。在圖表組件銷毀之前,務必使用
dispose
函數(shù)銷毀圖表實例,以確保釋放內(nèi)存。在動態(tài)更新圖表時,盡量復用現(xiàn)有的圖表實例,而不是每次都創(chuàng)建新的實例??梢允褂?Echarts 提供的
setOption
方法來更新圖表數(shù)據(jù)和配置。如果使用了定時器來控制圖表的刷新頻率,必須在組件銷毀前清定時器,可以使用
clearInterval
或clearTimeout
來停止定時器。正確管理圖表組件的生命周期,盡量避免不必要的渲染和更新操作。
初步驗證
按照以上的初步解決方案,我們對流程進行了初步優(yōu)化,主要進行了事件監(jiān)聽器的綁定與解綁優(yōu)化、銷毀定時器、dispose
函數(shù)銷毀圖表實例等操作。
等這些優(yōu)化操作完成后,我們又部署到自己的系統(tǒng)進行初步驗證,發(fā)現(xiàn)內(nèi)存占用確實變的小了,但是等到運行長時間來看,內(nèi)存還是有上升的趨勢。因此我斷定,沒有找到根本原因解決。
果然,運行了大概有五天的時間,瀏覽器還是頂不住了,系統(tǒng)再一次不工作了。
四. 第二輪解決方案
可能原因
由于這個大屏項目是其他同事開發(fā)的,具體的代碼我并不太清楚。因此,先前只是提供了一些建設性的建議,沒想到同事修改完成后還是沒有解決了根本問題。所以可能必須要完全讀懂項目的代碼才能找到根本原因。
果然,看了幾遍代碼后發(fā)現(xiàn)了一些端倪,頁面部署上之后是不會再進行操作了,數(shù)據(jù)會自動定時刷新,地圖數(shù)據(jù)也會自動刷新,因此,問題就出現(xiàn)在這了。
猜想
多次調(diào)用 Echarts.init,項目中這一段代碼寫的確實有問題,寫了個定時器,每次刷新數(shù)據(jù)時,都需要調(diào)用 Echarts init,并且銷毀時 clear 和 dispose 方法使用不當造成,定時器循環(huán)重繪 Echarts 圖表導致內(nèi)存一直升高,最終導致了瀏覽器崩潰。
解決方案
通過 Echarts init 方法創(chuàng)建 Echarts 實例,如果代碼沒有做優(yōu)化,echarts 實例就會越來越多,占用大量內(nèi)存,有以下兩種方法可以避免這種情況:
第一種:使用 Echarts init 之前先判斷是否存在實例
const chart = Echarts.getInstanceByDom(document.getElementById(dom)); if (chart === undefined) { chart = Echarts.init(document.getElementById(dom)); }
第二種:如果 Echarts 存在,先 dispose 銷毀后,再調(diào)用 init
const chart = Echarts.getInstanceByDom(document.getElementById(dom)); if (chart) { Echarts.dispose(chart); } chart = Echarts.init(document.getElementById(dom));
驗證
使用上述的代碼進行優(yōu)化,再結合第一輪的代碼優(yōu)化后。系統(tǒng)又重新部署了,監(jiān)測系統(tǒng)內(nèi)存狀態(tài),初始的內(nèi)存占用大小和之前的相差不大。不過觀察一段時間后,內(nèi)存沒有持續(xù)升高的趨勢,還算比較穩(wěn)定。又這樣運行了大概有一周左右,發(fā)現(xiàn)內(nèi)存占用仍然穩(wěn)定,系統(tǒng)也沒有出現(xiàn)過崩潰的問題。因此可以斷定,應該是優(yōu)化好了。
五. 總結
Echarts 圖表渲染導致的內(nèi)存泄漏問題是我們在使用 Echarts 時經(jīng)常遇到的挑戰(zhàn)之一。
在 Echarts 做圖表開發(fā)時,避免內(nèi)存泄漏的幾點操作主要有以下幾個方面:
多次調(diào)用 Echarts.init 會導致內(nèi)存泄漏,應當在恰當時機銷毀已經(jīng)存在的 Echarts 實例,使用 clear() 和 dispose() 手動清理,區(qū)別在于:
- clear()不會銷毀實例,只是重新繪制圖形,
- dispose()會銷毀實例,需要重新構建 ECharts 對象
在動態(tài)更新圖表時,盡量復用現(xiàn)有的圖表實例,而不是每次都創(chuàng)建新的實例??梢允褂?Echarts 提供的
setOption
方法來更新圖表數(shù)據(jù)和配置。在綁定事件監(jiān)聽器時,要確保正確解綁,可以使用 Echarts 提供的
off
方法來取消事件。
通過深入分析內(nèi)存泄漏的原因,并采取相應的解決方案,我們可以有效地解決這一問題,確保應用的穩(wěn)定性和性能。
以上就是淺析Echarts圖表渲染導致內(nèi)存泄漏的原因及解決方案的詳細內(nèi)容,更多關于Echarts圖表渲染導致內(nèi)存泄漏的資料請關注腳本之家其它相關文章!
相關文章
JavaScript中立即執(zhí)行函數(shù)實例詳解
javascript和其他編程語言相比比較隨意,所以javascript代碼中充滿各種奇葩的寫法,有時霧里看花,當然,能理解各型各色的寫法也是對javascript語言特性更進一步的深入理解。這篇文章主要給大家介紹了關于JavaScript中立即執(zhí)行函數(shù)的相關資料,需要的朋友可以參考下。2017-11-11javascript筆試題目附答案@20081025_jb51.net
網(wǎng)上找的javascript筆試題目,留檔給自己作參考。2008-10-10詳解如何準確判斷JavaScript中的數(shù)據(jù)類型
JavaScript中,我們經(jīng)常需要判斷數(shù)據(jù)類型以便于正確地處理數(shù)據(jù),本文將介紹JavaScript中的數(shù)據(jù)類型判斷技術,包括typeof操作符、instanceof操作符、Object.prototype.toString方法以及ES6新增的一些數(shù)據(jù)類型判斷方法,需要的朋友可以參考下2023-08-08