最近開發(fā)的時(shí)候?qū)撁媸褂昧硕〞r(shí)的局部更新,結(jié)果在ie6,7和Firefox下,一切正常,而在ie8下過上幾個(gè)小時(shí)就瀏覽器就崩潰了,顯示是內(nèi)存溢出,我以為是代碼寫的不好導(dǎo)致內(nèi)存泄露,但是ie6,7又正常,調(diào)查了一下,原來這是ie8的bug。
問題點(diǎn)
在IE8中,生成特定Dom節(jié)點(diǎn)所占用的內(nèi)存是不會(huì)被釋放的,即使這些節(jié)點(diǎn)被刪除內(nèi)存也不會(huì)被釋放。
內(nèi)存泄露的節(jié)點(diǎn)類型包括:form、button、input、select、textarea、a、img和objec
其他的大部分節(jié)點(diǎn)類型是不會(huì)泄露的,例如:span、div、p、table等等。
此問題只發(fā)生在IE8,其他瀏覽器不發(fā)生。
如果用戶按了F5,IE8會(huì)重新刷新頁面,首先它會(huì)unload window.top,這時(shí)候會(huì)釋放掉內(nèi)存。如果頁面是iframe,則unload此iframe,沒有任何反應(yīng)??雌饋碇挥衱indow.top被 unload,內(nèi)存才會(huì)被釋放。
例子:
例1
執(zhí)行下面的代碼,IE8就會(huì)泄露內(nèi)存。
function leak1() {
var node = document.getElementById("TO_AREA");
node.innerHTML = "<img />";
node.innerHTML = "";
node = null;
}
注意:
* 此例子添加了節(jié)點(diǎn),所以會(huì)泄露。
* 在中有個(gè)div,id為“TO_AREA”。
* 提醒一下,這里沒有閉包和循環(huán)引用。
例2
下面的代碼沒有使用innerHTML,但是還是會(huì)泄露
function leak2() {
var node = document.getElementById("FROM_AREA").cloneNode(true);
node.id = "NEW_AREA";
document.body.appendChild(node);
document.body.removeChild(node);
node = null;
}
注意:
* FROM_AREA 是form的id,而且這里也沒有閉包和循環(huán)引用。
例3
這是最簡(jiǎn)單,最直接的例子:
function leak4() {
var node = document.createElement("IMG");
document.body.appendChild(node);
document.body.removeChild(node);
}
注意:
* 如果用span來代替img,就不會(huì)有泄露了。
這些例子只在IE8中泄露內(nèi)存,我在Windows XP, Windows Vista, Windows Server 2008, Windows Server 2008 R2和Windows 7 中的IE8都作了測(cè)試,而且使用了IE8中的IE7兼容模式和標(biāo)準(zhǔn)模式,每種情況下都會(huì)泄露。
測(cè)試頁面
關(guān)于泄露
內(nèi)存大小隨著時(shí)間的推移而增長,但這并不直接導(dǎo)致瀏覽器崩潰。瀏覽器使用的內(nèi)存好像是有上限的,它似乎會(huì)從某些內(nèi)部手段來限制DHTML使用的內(nèi)存。
內(nèi)存到達(dá)上限后,瀏覽器會(huì)自動(dòng)處理,例如彈出對(duì)話框,顯示內(nèi)存不足。
經(jīng)過自己測(cè)試發(fā)現(xiàn) IFrame同樣存在這個(gè)問題(在IE8下)
補(bǔ)充:iframe內(nèi)存釋放
Ext 核心開發(fā)人員Jack的回答是,TabPanelItem在關(guān)閉時(shí)并不會(huì)對(duì)自定義到tab中的元素做特殊處理,這部分工作必須在控件外來完成。另一方面, 相關(guān)資料稱IE在iframe元素的回收方面存在著bug,在通常情況下應(yīng)該將該元素的src屬性值修改為”abort:blank”,并手工將其從 DOM樹上移除,然后把腳本中引用它的變量置空并調(diào)用CollectGarbage()就可以避免iframe不能正?;厥账斐傻膬?nèi)存泄露。
<script>
function clearRAM() {
var frame = document.getElementById("ifr_content");
frame.src = 'about:blank';
frame.contentWindow.document.write( '');//清空frame的內(nèi)容
frame.contentWindow.document.clear();
frame.contentWindow.close(); //避免frame內(nèi)存泄漏
if (navigator.userAgent.indexOf('MSIE') >= 0) {
if (CollectGarbage) {
CollectGarbage(); //IE 特有 釋放內(nèi)存
//刪除原有標(biāo)記
var tags = document.getElementById("ifrSet");
tags.removeChild(frame);
//添加frameset框架
var _frame = document.createElement('frame');
_frame.src = '';
_frame.name = 'content';
_frame.id = 'ifr_content';
tags.appendChild(_frame);
}
}
}
//主動(dòng)釋放 5秒一次
setInterval( function() {
if (navigator.userAgent.indexOf('MSIE') >= 0) {
if (CollectGarbage) {
//alert(1)
CollectGarbage(); //IE 特有 釋放內(nèi)存
}
}
}, 5000)
</ script>