JS常見內(nèi)存泄漏及解決方案解析
內(nèi)存泄漏?
官方解釋:內(nèi)存泄漏(Memory Leak)是指程序中己動(dòng)態(tài)分配的堆內(nèi)存由于某種原因程序未釋放或無法釋放,造成系統(tǒng)內(nèi)存的浪費(fèi),導(dǎo)致程序運(yùn)行速度減慢甚至系統(tǒng)崩潰等嚴(yán)重后果。
通俗點(diǎn)就是指由于疏忽或者錯(cuò)誤造成程序未能釋放已經(jīng)不再使用的內(nèi)存,不再用到的內(nèi)存卻沒有及時(shí)釋放,從而造成內(nèi)存上的浪費(fèi)。
避免內(nèi)存泄漏?
在局部作用域中,等函數(shù)執(zhí)行完畢,變量就沒有存在的必要了,垃圾回收機(jī)制很虧地做出判斷并且回收,但是對于全局變量,很難判斷什么時(shí)候不用這些變量,無法正常回收;所以,盡量少使用全局變量。在使用閉包的時(shí)候,就會(huì)造成嚴(yán)重的內(nèi)存泄漏,因?yàn)殚]包中的局部變量,會(huì)一直保存在內(nèi)存中。
內(nèi)存溢出?
當(dāng)程序運(yùn)行需要的內(nèi)存超過了剩余的內(nèi)存時(shí), 就出拋出內(nèi)存溢出的錯(cuò)誤。例如下面的代碼,謹(jǐn)慎試用,可能會(huì)卡窗口。。。。
var obj = {}for (var i = 0; i < 100000; i++) {obj[i] = new Array(10000000)}console.log(’------’)
常見的js內(nèi)存泄漏
1. 意外的全局變量
在js中,一個(gè)未聲明變量的使用,會(huì)在全局對象中創(chuàng)建一個(gè)新的變量;在瀏覽器環(huán)境下,全局對象就是window:
function foo() {a = ’test’}// 上面的寫法等價(jià)于function foo() {window.a = ’test’}
function foo() {this.a = ’test’// 函數(shù)自身發(fā)生調(diào)用,this指向全局對象window}foo();
上面的a變量應(yīng)該是foo()內(nèi)部作用域變量的引用,由于沒有使用var來聲明這個(gè)變量,這時(shí)變量a就被創(chuàng)建成了全局變量,這個(gè)就是錯(cuò)誤的,會(huì)導(dǎo)致內(nèi)存泄漏。
解決方式: 在js文件開頭添加 ‘use strict’,開啟嚴(yán)格模式。(或者一般將使用過后的全局變量設(shè)置為 null 或者將它重新賦值,這個(gè)會(huì)涉及的緩存的問題,需要注意)
<script> 'use strict'; console.log('這是嚴(yán)格模式。'); </script> <script> console.log('這是正常模式。'); </script>
2. 計(jì)時(shí)器和回調(diào)函數(shù)timers
定時(shí)器setInterval或者setTimeout在不需要使用的時(shí)候,沒有被clear,導(dǎo)致定時(shí)器的回調(diào)函數(shù)及其內(nèi)部依賴的變量都不能被回收,這就會(huì)造成內(nèi)存泄漏。
解決方式:當(dāng)不需要interval或者timeout的時(shí)候,調(diào)用clearInterval或者clearTimeout
3. DOM泄漏
1)給DOM對象添加的屬性是一個(gè)對象的引用
var a = {};document.getElementById(’id’).diyProp = a;
解決方法:在window.onload時(shí)間中加上 document.getElementById(’id’).diyProp = null;
2)元素引用沒有清理
var a = document.getElementById(’id’);document.body.removeChild(a);// 不能回收,因?yàn)榇嬖谧兞縜對它的引用。雖然我們用removeChild移除了,但是還在對象里保存著#的引用,即DOM元素還在內(nèi)存里面。
解決方法: a = null;
3)事件的綁定沒有移除
解決方法: 移除時(shí)間的監(jiān)聽
4. js閉包
閉包在IE6下會(huì)造成內(nèi)存泄漏,但是現(xiàn)在已經(jīng)無須考慮了。值得注意的是閉包本身不會(huì)造成內(nèi)存泄漏,但閉包過多很容易導(dǎo)致內(nèi)存泄漏。閉包會(huì)造成對象引用的生命周期脫離當(dāng)前函數(shù)的上下文,如果閉包如果使用不當(dāng),可以導(dǎo)致環(huán)形引用(circular reference),類似于死鎖,只能避免,無法發(fā)生之后解決,即使有垃圾回收也還是會(huì)內(nèi)存泄露。
這個(gè),另外找個(gè)時(shí)間詳細(xì)說明一下,這里就不贅述了
5. console
控制臺(tái)日志記錄對總體內(nèi)存內(nèi)置文件的影響,也是個(gè)重大的問題,同時(shí)也是容易被忽略的。記錄錯(cuò)誤的對象,可以將大量的數(shù)據(jù)保留在內(nèi)存中。傳遞給console.log的對象是不能被垃圾回收,所以沒有去掉console.log可能會(huì)存在內(nèi)存泄漏
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持好吧啦網(wǎng)。
相關(guān)文章:
1. .NET中l(wèi)ambda表達(dá)式合并問題及解決方法2. JSP數(shù)據(jù)交互實(shí)現(xiàn)過程解析3. 淺談python出錯(cuò)時(shí)traceback的解讀4. 利用promise及參數(shù)解構(gòu)封裝ajax請求的方法5. Python importlib動(dòng)態(tài)導(dǎo)入模塊實(shí)現(xiàn)代碼6. python matplotlib:plt.scatter() 大小和顏色參數(shù)詳解7. windows服務(wù)器使用IIS時(shí)thinkphp搜索中文無效問題8. ASP 信息提示函數(shù)并作返回或者轉(zhuǎn)向9. 在Android中使用WebSocket實(shí)現(xiàn)消息通信的方法詳解10. Nginx+php配置文件及原理解析
