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