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