詳解JavaScript如何避免內(nèi)存泄漏
前言
過去,我們?yōu)g覽靜態(tài)網(wǎng)站時(shí)無須過多關(guān)注內(nèi)存管理,因?yàn)榧虞d新頁面時(shí),之前的頁面信息會(huì)從內(nèi)存中刪除。 然而,隨著單頁Web應(yīng)用(SPA)的興起,應(yīng)用程序消耗的內(nèi)存越來越多,這不僅會(huì)降低瀏覽器性能,甚至?xí)?dǎo)致瀏覽器卡死。因此,在編碼實(shí)踐中,開發(fā)人員需要更加關(guān)注與內(nèi)存相關(guān)的內(nèi)容。因此,小編今天將為大家介紹JavaScript內(nèi)存泄漏的編程模式,并提供一些內(nèi)存管理的改進(jìn)方法。
什么是內(nèi)存泄漏以及如何發(fā)現(xiàn)它
什么是內(nèi)存泄漏
JavaScript對(duì)象被保存在瀏覽器內(nèi)存的堆中,并通過引用方式訪問。值得一提的是,JavaScript垃圾回收器則運(yùn)行于后臺(tái),并通過識(shí)別無法訪問的對(duì)象來釋放并恢復(fù)底層存儲(chǔ)空間,從而保證JavaScript引擎的良好運(yùn)行狀態(tài)。
當(dāng)內(nèi)存中的對(duì)象在垃圾回收周期中應(yīng)該被清理時(shí),若它們被另一個(gè)仍然存在于內(nèi)存中的對(duì)象通過一個(gè)意外的引用所持有,就會(huì)引發(fā)內(nèi)存泄漏問題。這種情況下,冗余對(duì)象會(huì)繼續(xù)占據(jù)內(nèi)存空間,導(dǎo)致應(yīng)用程序消耗過多的內(nèi)存資源,并可能導(dǎo)致性能下降和表現(xiàn)不佳的情況出現(xiàn)。因此,及時(shí)清理無用對(duì)象并釋放內(nèi)存資源是至關(guān)重要的,以確保應(yīng)用程序的正常運(yùn)行和良好的性能表現(xiàn)。
如何發(fā)現(xiàn)內(nèi)存泄漏
那么如何知道代碼中是否存在內(nèi)存泄漏??jī)?nèi)存泄漏往往隱蔽且很難檢測(cè)和定位。即使代碼中存在內(nèi)存泄漏,瀏覽器在運(yùn)行時(shí)也不會(huì)返回任何錯(cuò)誤。如果注意到頁面的性能逐漸下降,可以使用瀏覽器內(nèi)置的工具來確定是否存在內(nèi)存泄漏以及是哪個(gè)對(duì)象引起的。
任務(wù)管理器(不要與操作系統(tǒng)的任務(wù)管理器混淆)提供了瀏覽器中所有選項(xiàng)卡和進(jìn)程的概覽。Chrome 中,可以通過在 Linux 和 Windows 操作系統(tǒng)上按 Shift+Esc 來打開任務(wù)管理器;而在 Firefox 中,通過在地址欄中鍵入 about:performance 則可以訪問內(nèi)置的管理器,它可以顯示每個(gè)標(biāo)簽的 JavaScript 內(nèi)存占用情況。如果網(wǎng)站停留在那里什么都不做,但 JavaScript內(nèi)存使用量逐漸增加,那很可能是存在內(nèi)存泄漏。
開發(fā)者工具提供了一些先進(jìn)的內(nèi)存管理方法,例如,使用Chrome瀏覽器的性能記錄工具,可以對(duì)頁面的性能進(jìn)行可視化分析。在這個(gè)過程中,可以通過一些指標(biāo)來判斷是否存在內(nèi)存泄漏問題,比如堆內(nèi)存使用量增加的情況,并及時(shí)采取措施解決這些問題,以確保應(yīng)用程序的正常運(yùn)行和良好的性能表現(xiàn)。
另外,通過Chrome和Firefox的開發(fā)者工具提供的內(nèi)存工具,可以進(jìn)一步探索內(nèi)存使用情況。隊(duì)列內(nèi)存使用快照的比較可以顯示在兩個(gè)快照之間分配了多少內(nèi)存以及分配的位置,并提供額外信息來幫助識(shí)別代碼中存在問題的對(duì)象。這些工具為開發(fā)者提供了便利,能夠更好地進(jìn)行內(nèi)存管理和性能優(yōu)化,提高應(yīng)用程序的質(zhì)量和性能。
JavaScript代碼中常見的內(nèi)存泄漏的常見來源
研究?jī)?nèi)存泄漏問題就相當(dāng)于尋找符合垃圾回收機(jī)制的編程方式,有效避免對(duì)象引用的問題。下面小編就為大家介紹幾個(gè)常見的容易導(dǎo)致內(nèi)存泄漏的地方:
1.全局變量
全局變量始終存儲(chǔ)在根目錄下,且永遠(yuǎn)不會(huì)被回收。而在JavaScript的開發(fā)中,一些錯(cuò)誤會(huì)導(dǎo)致局部變量被轉(zhuǎn)換到了全局,尤其是在非嚴(yán)格的代碼模式下。下面是兩個(gè)常見的局部變量被轉(zhuǎn)化到全局變量的情況:
- 為未聲明的變量賦值
- 使用this指向全局對(duì)象。
function createGlobalVariables() { leaking1 = 'I leak into the global scope'; // 為未聲明的變量賦值 this.leaking2 = 'I also leak into the global scope'; // 使用this指向全局對(duì)象 }; createGlobalVariables(); window.leaking1; window.leaking2;
注意:嚴(yán)格模式("use strict")將幫助您避免上面示例中的內(nèi)存泄漏和控制臺(tái)錯(cuò)誤。
2.閉包
函數(shù)中定義的變量會(huì)在函數(shù)退出調(diào)用棧并且在函數(shù)外部沒有指向它的引用時(shí)被清除。而閉包則會(huì)保持被引用的變量一直存在,即便函數(shù)的執(zhí)行已經(jīng)終止。
function outer() { const potentiallyHugeArray = []; return function inner() { potentiallyHugeArray.push('Hello'); // function inner is closed over the potentiallyHugeArray variable console.log('Hello'); }; }; const sayHello = outer(); // contains definition of the function inner function repeat(fn, num) { for (let i = 0; i < num; i++){ fn(); } } repeat(sayHello, 10); // each sayHello call pushes another 'Hello' to the potentiallyHugeArray // now imagine repeat(sayHello, 100000)
在這個(gè)例子中,potentiallyHugeArray從未被任何函數(shù)返回,也無法被訪問,但它的大小會(huì)隨著調(diào)用 inner 方法的次數(shù)而增長(zhǎng)。
3.定時(shí)器
在JavaScript中,使用使用 setTimeout 或 setInterval函數(shù)引用對(duì)象是防止對(duì)象被垃圾回收的最常見方法。當(dāng)在代碼中設(shè)置循環(huán)定時(shí)器(可以使 setTimeout 表現(xiàn)得像 setInterval,即使其遞歸)時(shí),只要回調(diào)可調(diào)用,定時(shí)器回調(diào)對(duì)象的引用就會(huì)永遠(yuǎn)保持活動(dòng)狀態(tài)。
例如下面的這段代碼,只有在移除定時(shí)器后,data對(duì)象才會(huì)被垃圾回收。在沒有移除setInterval之前,它永遠(yuǎn)不會(huì)被刪除,并且data.hugeString 會(huì)一直保留在內(nèi)存中,直到應(yīng)用程序停止。
function setCallback() { const data = { counter: 0, hugeString: new Array(100000).join('x') }; return function cb() { data.counter++; // data object is now part of the callback's scope console.log(data.counter); } } setInterval(setCallback(), 1000); // how do we stop it?
那么應(yīng)該如何避免上述這種情況的發(fā)生呢?可以從以下兩個(gè)方法入手:
- 注意定時(shí)器回調(diào)引用的對(duì)象。
- 必要時(shí)取消定時(shí)器。
如下方的代碼所示:
function setCallback() { // 'unpacking' the data object let counter = 0; const hugeString = new Array(100000).join('x'); // gets removed when the setCallback returns return function cb() { counter++; // only counter is part of the callback's scope console.log(counter); } } const timerId = setInterval(setCallback(), 1000); // saving the interval ID // doing something ... clearInterval(timerId); // stopping the timer i.e. if button pressed
4.事件監(jiān)聽
活動(dòng)的事件監(jiān)聽器會(huì)阻止其范圍內(nèi)的所有變量被回收。一旦添加,事件監(jiān)聽器會(huì)一直生效,直到下面兩種情況的發(fā)生:
- 通過 removeEventListener() 移除。
- 相關(guān)聯(lián)的 DOM 元素被移除。
在下面的示例中,使用匿名內(nèi)聯(lián)函數(shù)作為事件監(jiān)聽器,這意味著它不能與 removeEventListener() 一起使用。此外,由于document 不能被移除,觸發(fā)方法中的內(nèi)容會(huì)一直駐留內(nèi)存,即使只使用它觸發(fā)一次。
const hugeString = new Array(100000).join('x'); document.addEventListener('keyup', function() { // anonymous inline function - can't remove it doSomething(hugeString); // hugeString is now forever kept in the callback's scope });
那么如何避免這種情況呢?可以通過removeEventListener()釋放監(jiān)聽器:
function listener() { doSomething(hugeString); } document.addEventListener('keyup', listener); // named function can be referenced here... document.removeEventListener('keyup', listener); // ...and here
如果事件監(jiān)聽器只需要運(yùn)行一次,addEventListener() 可以帶有第三個(gè)參數(shù),一個(gè)提供附加選項(xiàng)的對(duì)象。只要將 {once: true} 作為第三個(gè)參數(shù)傳遞給 addEventListener(),監(jiān)聽器將在事件處理一次后自動(dòng)刪除。
document.addEventListener('keyup', function listener() { doSomething(hugeString); }, {once: true}); // listener will be removed after running once
5.緩存
如果不斷向緩存中添加內(nèi)容,而未使用的對(duì)象也沒有移除,也沒有限制緩存的大小,那么緩存的大小就會(huì)無限增長(zhǎng):
let user_1 = { name: "Peter", id: 12345 }; let user_2 = { name: "Mark", id: 54321 }; const mapCache = new Map(); function cache(obj){ if (!mapCache.has(obj)){ const value = `${obj.name} has an id of ${obj.id}`; mapCache.set(obj, value); return [value, 'computed']; } return [mapCache.get(obj), 'cached']; } cache(user_1); // ['Peter has an id of 12345', 'computed'] cache(user_1); // ['Peter has an id of 12345', 'cached'] cache(user_2); // ['Mark has an id of 54321', 'computed'] console.log(mapCache); // ((…) => "Peter has an id of 12345", (…) => "Mark has an id of 54321") user_1 = null; // removing the inactive user // Garbage Collector console.log(mapCache); // ((…) => "Peter has an id of 12345", (…) => "Mark has an id of 54321") // first entry is still in cache
為了解決這個(gè)問題,需要清除不需要的緩存:
一種有效的解決內(nèi)存泄漏問題的方法是使用WeakMap。它是一種數(shù)據(jù)結(jié)構(gòu),其中鍵引用被保持為弱引用,并且僅接受對(duì)象作為鍵。如果使用對(duì)象作為鍵,并且它是唯一引用該對(duì)象的引用,相關(guān)條目將從緩存中移除,并進(jìn)行垃圾回收。在下面的示例中,當(dāng)替換user_1后,與之關(guān)聯(lián)的條目將在下一次垃圾回收時(shí)自動(dòng)從WeakMap中移除。
let user_1 = { name: "Peter", id: 12345 }; let user_2 = { name: "Mark", id: 54321 }; const weakMapCache = new WeakMap(); function cache(obj){ // ...same as above, but with weakMapCache return [weakMapCache.get(obj), 'cached']; } cache(user_1); // ['Peter has an id of 12345', 'computed'] cache(user_2); // ['Mark has an id of 54321', 'computed'] console.log(weakMapCache); // ((…) => "Peter has an id of 12345", (…) => "Mark has an id of 54321"} user_1 = null; // removing the inactive user // Garbage Collector console.log(weakMapCache); // ((…) => "Mark has an id of 54321") - first entry gets garbage collected
結(jié)論
對(duì)于復(fù)雜的應(yīng)用程序,檢測(cè)和修復(fù) JavaScript 內(nèi)存泄漏問題可能是一項(xiàng)非常艱巨的任務(wù)。了解內(nèi)存泄漏的常見原因以防止它們發(fā)生是非常重要的。在涉及內(nèi)存和性能方面,最重要的是用戶體驗(yàn),這才是最重要的。
到此這篇關(guān)于詳解JavaScript如何避免內(nèi)存泄漏的文章就介紹到這了,更多相關(guān)JavaScript內(nèi)存泄漏內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
微信小程序中如何實(shí)現(xiàn)動(dòng)態(tài)改變SVG顏色和尺寸
SVG可被非常多的工具讀取和修改SVG與JPEG和GIF圖像比起來,尺寸更小,且可壓縮性更強(qiáng),下面這篇文章主要給大家介紹了關(guān)于微信小程序中如何實(shí)現(xiàn)動(dòng)態(tài)改變SVG顏色和尺寸的相關(guān)資料,需要的朋友可以參考下2022-07-07僅9張思維導(dǎo)圖幫你輕松學(xué)習(xí)Javascript 就這么簡(jiǎn)單
僅9張思維導(dǎo)圖幫你輕松學(xué)習(xí)Javascript,從javascript變量、javascript運(yùn)算符、javascript函數(shù)基礎(chǔ)等多方面了解Javascript,就這么簡(jiǎn)單2016-06-06js如何實(shí)現(xiàn)設(shè)計(jì)模式中的模板方法
都知道在js中如果定義兩個(gè)相同名稱的方法,前一個(gè)方法就會(huì)被后一個(gè)方法覆蓋掉,使用此特點(diǎn)就可以實(shí)現(xiàn)模板方法,感興趣的朋友可以了解下本文哈2013-07-07html中通過JS獲取JSON數(shù)據(jù)并加載的方法
本篇內(nèi)容主要給大家講了如何通過javascript解析JSON并得到數(shù)據(jù)后添加到HTML中的方法,需要的朋友參考下。2017-11-11js實(shí)現(xiàn)圖片放大并跟隨鼠標(biāo)移動(dòng)特效
這篇文章主要為大家詳細(xì)介紹了js實(shí)現(xiàn)圖片放大并跟隨鼠標(biāo)移動(dòng)特效,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2019-01-01通過JS判斷聯(lián)網(wǎng)類型和連接狀態(tài)的實(shí)現(xiàn)代碼
這篇文章主要介紹了通過JS判斷聯(lián)網(wǎng)類型和連接狀態(tài)的實(shí)現(xiàn)代碼,需要的朋友可以參考下2015-04-04javascript在事件監(jiān)聽方面的兼容性小結(jié)
javascript 在事件監(jiān)聽方面的兼容性總結(jié),注意是由于多個(gè)瀏覽器的不一致,導(dǎo)致大家在js書寫時(shí)需要考慮多個(gè)瀏覽器的兼容性。2010-04-04