淺析Echarts圖表渲染導(dǎo)致內(nèi)存泄漏的原因及解決方案
一. 引言
在今年某個可視化大屏項目中,出現(xiàn)了一個問題。項目在運(yùn)行一段時間后,頁面出現(xiàn)了崩潰,而且是大概運(yùn)行幾天之后,因?yàn)榇笃另椖渴遣渴鸬娇蛻衄F(xiàn)場大屏,長時間運(yùn)行不關(guān)閉。報錯問題如下圖所示:

由于這個大屏頁面是使用 Echarts 圖表渲染的,整個頁面就一個渲染地圖的功能,沒有多少耗費(fèi)內(nèi)存的操作。而 Echarts 作為一款強(qiáng)大的數(shù)據(jù)可視化庫,被廣泛應(yīng)用于各種項目中,因此 Echarts 的穩(wěn)定性也是經(jīng)過廣大開發(fā)者驗(yàn)證的,不會出現(xiàn)明顯的 bug 問題。
然而,我們也知道,如果頁面隨著圖表數(shù)量的增加和動態(tài)更新的需求,我們必然會考慮 Echarts 圖表渲染導(dǎo)致的內(nèi)存泄漏問題。因此,上面的問題我們按照這個猜想來往下走,認(rèn)為 Echarts 圖表渲染導(dǎo)致了內(nèi)存泄漏,那么 Echarts 的什么操作會導(dǎo)致內(nèi)存泄漏、甚至到頁面崩潰呢?
本文將深入分析這一問題,并提供解決方案。
二. 問題背景
猜測原因
當(dāng)我們使用 Echarts 來渲染大量圖表時,會發(fā)現(xiàn)頁面的內(nèi)存占用不斷增加,最終導(dǎo)致頁面卡頓甚至崩潰。這是由于 Echarts 在圖表渲染過程中產(chǎn)生的內(nèi)存泄漏導(dǎo)致的。
而奇怪的是,自己在本地中測試的時候,從未有過崩潰的現(xiàn)象?但是在客戶現(xiàn)場就出現(xiàn)了這種問題。

內(nèi)存占用檢測
為了能夠?qū)崟r監(jiān)測網(wǎng)站運(yùn)行過程中,我們在 Chrome 瀏覽器打開了內(nèi)存性能分析,便于實(shí)時查看內(nèi)存占用情況,發(fā)現(xiàn)運(yùn)行內(nèi)存占用并不是太大,我們監(jiān)測過一段時間后發(fā)現(xiàn),確實(shí)內(nèi)存有時在穩(wěn)定的增加,但是這點(diǎn)內(nèi)存占用還不至于導(dǎo)致系統(tǒng)崩潰。

內(nèi)存泄漏原因分析
嵌套事件綁定:在 Echarts 圖表中,每個圖表都會綁定各種交互事件。如果我們正確地解綁這些事件,則會導(dǎo)致事件監(jiān)聽器無法垃圾回收,從而造成內(nèi)存泄漏。
大量的圖表實(shí)例:如果我們在動態(tài)更新圖表的過程中,每次都創(chuàng)建新的圖表實(shí)例而不銷毀舊的實(shí)例,就會導(dǎo)致內(nèi)存占用不斷增加。
定時器未清理:如果我們在更新圖表的過程中使用了定時器來控制刷新頻率,但未能正確清理這些定時器,就可能導(dǎo)致內(nèi)存泄漏。
三. 第一輪解決方案
因?yàn)榭梢暬笃另椖看a不是我直接寫的,我當(dāng)時只不過是被臨時抽過去提過幾個建議而已,所以并沒有深入的研讀項目代碼,所以這一輪解決方案可以定義為是在浪費(fèi)時間,不過也并不是一點(diǎn)經(jīng)驗(yàn)沒有,一起來看一下吧。
初步解決
為了解決 Echarts 圖表渲染導(dǎo)致的內(nèi)存泄漏問題,我們采取以下措施:
在綁定事件監(jiān)聽器時,要確保正確解綁,可以使用 Echarts 提供的
off方法來取消事件定。在圖表組件銷毀之前,務(wù)必使用
dispose函數(shù)銷毀圖表實(shí)例,以確保釋放內(nèi)存。在動態(tài)更新圖表時,盡量復(fù)用現(xiàn)有的圖表實(shí)例,而不是每次都創(chuàng)建新的實(shí)例。可以使用 Echarts 提供的
setOption方法來更新圖表數(shù)據(jù)和配置。如果使用了定時器來控制圖表的刷新頻率,必須在組件銷毀前清定時器,可以使用
clearInterval或clearTimeout來停止定時器。正確管理圖表組件的生命周期,盡量避免不必要的渲染和更新操作。
初步驗(yàn)證
按照以上的初步解決方案,我們對流程進(jìn)行了初步優(yōu)化,主要進(jìn)行了事件監(jiān)聽器的綁定與解綁優(yōu)化、銷毀定時器、dispose 函數(shù)銷毀圖表實(shí)例等操作。
等這些優(yōu)化操作完成后,我們又部署到自己的系統(tǒng)進(jìn)行初步驗(yàn)證,發(fā)現(xiàn)內(nèi)存占用確實(shí)變的小了,但是等到運(yùn)行長時間來看,內(nèi)存還是有上升的趨勢。因此我斷定,沒有找到根本原因解決。
果然,運(yùn)行了大概有五天的時間,瀏覽器還是頂不住了,系統(tǒng)再一次不工作了。
四. 第二輪解決方案
可能原因
由于這個大屏項目是其他同事開發(fā)的,具體的代碼我并不太清楚。因此,先前只是提供了一些建設(shè)性的建議,沒想到同事修改完成后還是沒有解決了根本問題。所以可能必須要完全讀懂項目的代碼才能找到根本原因。
果然,看了幾遍代碼后發(fā)現(xiàn)了一些端倪,頁面部署上之后是不會再進(jìn)行操作了,數(shù)據(jù)會自動定時刷新,地圖數(shù)據(jù)也會自動刷新,因此,問題就出現(xiàn)在這了。
猜想
多次調(diào)用 Echarts.init,項目中這一段代碼寫的確實(shí)有問題,寫了個定時器,每次刷新數(shù)據(jù)時,都需要調(diào)用 Echarts init,并且銷毀時 clear 和 dispose 方法使用不當(dāng)造成,定時器循環(huán)重繪 Echarts 圖表導(dǎo)致內(nèi)存一直升高,最終導(dǎo)致了瀏覽器崩潰。
解決方案
通過 Echarts init 方法創(chuàng)建 Echarts 實(shí)例,如果代碼沒有做優(yōu)化,echarts 實(shí)例就會越來越多,占用大量內(nèi)存,有以下兩種方法可以避免這種情況:
第一種:使用 Echarts init 之前先判斷是否存在實(shí)例
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àn)證
使用上述的代碼進(jìn)行優(yōu)化,再結(jié)合第一輪的代碼優(yōu)化后。系統(tǒng)又重新部署了,監(jiān)測系統(tǒng)內(nèi)存狀態(tài),初始的內(nèi)存占用大小和之前的相差不大。不過觀察一段時間后,內(nèi)存沒有持續(xù)升高的趨勢,還算比較穩(wěn)定。又這樣運(yùn)行了大概有一周左右,發(fā)現(xiàn)內(nèi)存占用仍然穩(wěn)定,系統(tǒng)也沒有出現(xiàn)過崩潰的問題。因此可以斷定,應(yīng)該是優(yōu)化好了。
五. 總結(jié)
Echarts 圖表渲染導(dǎo)致的內(nèi)存泄漏問題是我們在使用 Echarts 時經(jīng)常遇到的挑戰(zhàn)之一。
在 Echarts 做圖表開發(fā)時,避免內(nèi)存泄漏的幾點(diǎn)操作主要有以下幾個方面:
多次調(diào)用 Echarts.init 會導(dǎo)致內(nèi)存泄漏,應(yīng)當(dāng)在恰當(dāng)時機(jī)銷毀已經(jīng)存在的 Echarts 實(shí)例,使用 clear() 和 dispose() 手動清理,區(qū)別在于:
- clear()不會銷毀實(shí)例,只是重新繪制圖形,
- dispose()會銷毀實(shí)例,需要重新構(gòu)建 ECharts 對象
在動態(tài)更新圖表時,盡量復(fù)用現(xiàn)有的圖表實(shí)例,而不是每次都創(chuàng)建新的實(shí)例??梢允褂?Echarts 提供的
setOption方法來更新圖表數(shù)據(jù)和配置。在綁定事件監(jiān)聽器時,要確保正確解綁,可以使用 Echarts 提供的
off方法來取消事件。
通過深入分析內(nèi)存泄漏的原因,并采取相應(yīng)的解決方案,我們可以有效地解決這一問題,確保應(yīng)用的穩(wěn)定性和性能。
以上就是淺析Echarts圖表渲染導(dǎo)致內(nèi)存泄漏的原因及解決方案的詳細(xì)內(nèi)容,更多關(guān)于Echarts圖表渲染導(dǎo)致內(nèi)存泄漏的資料請關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
JavaScript中獲取隨機(jī)數(shù)的幾種方法小結(jié)
本文總結(jié)了JavaScript中獲取隨機(jī)數(shù)的幾種方法,包括Math.random()、生成指定范圍的隨機(jī)數(shù)和從數(shù)組中隨機(jī)選擇一個元素,具有一定的參考價值,感興趣的可以了解一下2025-02-02
使用gulp搭建本地服務(wù)器并實(shí)現(xiàn)模擬ajax
這篇文章主要給大家介紹了如何使用gulp搭建本地服務(wù)器并實(shí)現(xiàn)模擬ajax的相關(guān)資料,文中介紹的非常詳細(xì),相信對大家具有一定的參考學(xué)習(xí)價值,需要的朋友們下面來一起看看吧。2017-04-04
js中的數(shù)組轉(zhuǎn)樹型結(jié)構(gòu)方式
這篇文章主要介紹了js中的數(shù)組轉(zhuǎn)樹型結(jié)構(gòu)方式,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2023-06-06
javascript常用的數(shù)組過濾和查找方法總結(jié)
這篇文章主要介紹了javascript常用的數(shù)組過濾和查找方法的相關(guān)資料,這些方法各有特點(diǎn),適用于不同的數(shù)據(jù)處理場景,大家可以根據(jù)需求選擇不同的方法,需要的朋友可以參考下2024-12-12
JavaScript中立即執(zhí)行函數(shù)實(shí)例詳解
javascript和其他編程語言相比比較隨意,所以javascript代碼中充滿各種奇葩的寫法,有時霧里看花,當(dāng)然,能理解各型各色的寫法也是對javascript語言特性更進(jìn)一步的深入理解。這篇文章主要給大家介紹了關(guān)于JavaScript中立即執(zhí)行函數(shù)的相關(guān)資料,需要的朋友可以參考下。2017-11-11
javascript筆試題目附答案@20081025_jb51.net
網(wǎng)上找的javascript筆試題目,留檔給自己作參考。2008-10-10
詳解如何準(zhǔn)確判斷JavaScript中的數(shù)據(jù)類型
JavaScript中,我們經(jīng)常需要判斷數(shù)據(jù)類型以便于正確地處理數(shù)據(jù),本文將介紹JavaScript中的數(shù)據(jù)類型判斷技術(shù),包括typeof操作符、instanceof操作符、Object.prototype.toString方法以及ES6新增的一些數(shù)據(jù)類型判斷方法,需要的朋友可以參考下2023-08-08

