JS常見(jiàn)內(nèi)存泄漏及解決方案解析
內(nèi)存泄漏?
官方解釋:內(nèi)存泄漏(Memory Leak)是指程序中己動(dòng)態(tài)分配的堆內(nèi)存由于某種原因程序未釋放或無(wú)法釋放,造成系統(tǒng)內(nèi)存的浪費(fèi),導(dǎo)致程序運(yùn)行速度減慢甚至系統(tǒng)崩潰等嚴(yán)重后果。
通俗點(diǎn)就是指由于疏忽或者錯(cuò)誤造成程序未能釋放已經(jīng)不再使用的內(nèi)存,不再用到的內(nèi)存卻沒(méi)有及時(shí)釋放,從而造成內(nèi)存上的浪費(fèi)。
避免內(nèi)存泄漏?
在局部作用域中,等函數(shù)執(zhí)行完畢,變量就沒(méi)有存在的必要了,垃圾回收機(jī)制很虧地做出判斷并且回收,但是對(duì)于全局變量,很難判斷什么時(shí)候不用這些變量,無(wú)法正?;厥?;所以,盡量少使用全局變量。在使用閉包的時(shí)候,就會(huì)造成嚴(yán)重的內(nèi)存泄漏,因?yàn)殚]包中的局部變量,會(huì)一直保存在內(nèi)存中。
內(nèi)存溢出?
當(dāng)程序運(yùn)行需要的內(nèi)存超過(guò)了剩余的內(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(’------’)
常見(jiàn)的js內(nèi)存泄漏
1. 意外的全局變量
在js中,一個(gè)未聲明變量的使用,會(huì)在全局對(duì)象中創(chuàng)建一個(gè)新的變量;在瀏覽器環(huán)境下,全局對(duì)象就是window:
function foo() {a = ’test’}// 上面的寫(xiě)法等價(jià)于function foo() {window.a = ’test’}
function foo() {this.a = ’test’// 函數(shù)自身發(fā)生調(diào)用,this指向全局對(duì)象window}foo();
上面的a變量應(yīng)該是foo()內(nèi)部作用域變量的引用,由于沒(méi)有使用var來(lái)聲明這個(gè)變量,這時(shí)變量a就被創(chuàng)建成了全局變量,這個(gè)就是錯(cuò)誤的,會(huì)導(dǎo)致內(nèi)存泄漏。
解決方式: 在js文件開(kāi)頭添加 ‘use strict’,開(kāi)啟嚴(yán)格模式。(或者一般將使用過(guò)后的全局變量設(shè)置為 null 或者將它重新賦值,這個(gè)會(huì)涉及的緩存的問(wèn)題,需要注意)
<script> 'use strict'; console.log('這是嚴(yán)格模式。'); </script> <script> console.log('這是正常模式。'); </script>
2. 計(jì)時(shí)器和回調(diào)函數(shù)timers
定時(shí)器setInterval或者setTimeout在不需要使用的時(shí)候,沒(méi)有被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對(duì)象添加的屬性是一個(gè)對(duì)象的引用
var a = {};document.getElementById(’id’).diyProp = a;
解決方法:在window.onload時(shí)間中加上 document.getElementById(’id’).diyProp = null;
2)元素引用沒(méi)有清理
var a = document.getElementById(’id’);document.body.removeChild(a);// 不能回收,因?yàn)榇嬖谧兞縜對(duì)它的引用。雖然我們用removeChild移除了,但是還在對(duì)象里保存著#的引用,即DOM元素還在內(nèi)存里面。
解決方法: a = null;
3)事件的綁定沒(méi)有移除
解決方法: 移除時(shí)間的監(jiān)聽(tīng)
4. js閉包
閉包在IE6下會(huì)造成內(nèi)存泄漏,但是現(xiàn)在已經(jīng)無(wú)須考慮了。值得注意的是閉包本身不會(huì)造成內(nèi)存泄漏,但閉包過(guò)多很容易導(dǎo)致內(nèi)存泄漏。閉包會(huì)造成對(duì)象引用的生命周期脫離當(dāng)前函數(shù)的上下文,如果閉包如果使用不當(dāng),可以導(dǎo)致環(huán)形引用(circular reference),類似于死鎖,只能避免,無(wú)法發(fā)生之后解決,即使有垃圾回收也還是會(huì)內(nèi)存泄露。
這個(gè),另外找個(gè)時(shí)間詳細(xì)說(shuō)明一下,這里就不贅述了
5. console
控制臺(tái)日志記錄對(duì)總體內(nèi)存內(nèi)置文件的影響,也是個(gè)重大的問(wèn)題,同時(shí)也是容易被忽略的。記錄錯(cuò)誤的對(duì)象,可以將大量的數(shù)據(jù)保留在內(nèi)存中。傳遞給console.log的對(duì)象是不能被垃圾回收,所以沒(méi)有去掉console.log可能會(huì)存在內(nèi)存泄漏
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持好吧啦網(wǎng)。
相關(guān)文章:
1. jsp實(shí)現(xiàn)簡(jiǎn)單用戶7天內(nèi)免登錄2. ASP基礎(chǔ)入門第二篇(ASP基礎(chǔ)知識(shí))3. ASP中Server.HTMLEncode用法(附自定義函數(shù))4. ASP和PHP文件操作速度的對(duì)比5. ASP替換、保存遠(yuǎn)程圖片實(shí)現(xiàn)代碼6. adodb.recordset.open(rs.open)方法參數(shù)詳解7. jsp實(shí)現(xiàn)局部刷新頁(yè)面、異步加載頁(yè)面的方法8. 怎樣打開(kāi)XML文件?xml文件如何打開(kāi)?9. Spring依賴注入的三種方式實(shí)例詳解10. asp文件如何打開(kāi)
