IE9之前的版本對(duì)JScript對(duì)象和COM對(duì)象使用不同的垃圾回收例程(COM對(duì)象采用“引用計(jì)數(shù)”收集策略),因此閉包在IE的這些版本中會(huì)導(dǎo)致一些特殊問(wèn)題。具體來(lái)說(shuō),如果閉包的作用域中保存著一個(gè)HTML元素,那么就意味著該元素將無(wú)法被銷毀。
來(lái)看下面的例子:
function assignHandler() { var elem = document.getElementById('elem_id'); elem.onclick = function(evt) { alert(elem.id); };}以上代碼創(chuàng)建了一個(gè)作為elem元素事件處理程序的閉包,而這個(gè)閉包則又創(chuàng)建了一個(gè)循環(huán)引用。由于匿名函數(shù)保存了一個(gè)對(duì)assignHandler()的活動(dòng)對(duì)象的引用,因此就會(huì)導(dǎo)致無(wú)法減少elem的引用數(shù)。只要匿名函數(shù)存在,elem的引用數(shù)至少也是1,因此它所占用的內(nèi)存就永遠(yuǎn)不會(huì)被回收。
可以將上面的代碼稍作修改一下就可以解決:
function assignHandler() { var elem = document.getElementById('elem_id'); var elem_id = elem.id; elem.onclick = function(evt) { alert(elem_id); }; elem = null;}通過(guò)把elem.id的一個(gè)副本保存在一個(gè)變量中,并且在閉包中引用該變量消除了循環(huán)引用。但僅僅做到這一步,還是不能解決內(nèi)存泄露問(wèn)題。
“閉包會(huì)引用包含函數(shù)的整個(gè)活動(dòng)對(duì)象,而其中就包含著elem。即使閉包不直接引用elem,包含函數(shù)的活動(dòng)對(duì)象中也仍然會(huì)保存一個(gè)引用。因此,有必要把elem設(shè)置為null。這樣就能解除對(duì)DOM對(duì)象的引用,順利地減少其引用數(shù),確保正?;厥掌湔加玫膬?nèi)存”
新聞熱點(diǎn)
疑難解答
圖片精選