為得求證,自己寫了一個頁面來驗證怎樣內存洩漏。代碼如下
復制代碼 代碼如下:
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">
<html xmlns="http://www.w3.org/1999/xhtml">
<head>
<meta http-equiv="Content-Type" content="text/html; charset=utf-8" />
<title>測試內存洩漏</title>
<script type="text/javascript">
function creatDiv()
{
var divObj = document.createElement("div");
divObj.id="testDiv";
divObj.innerHTML = "用來測試的DIV";
document.getElementById("main").appendChild(divObj);
}
function removeDiv()
{
document.getElementById("main").removeChild(document.getElementById("testDiv"));
}
function checkDiv()
{
alert(document.getElementById("testDiv"));
}
</script>
</head>
<body>
<div id="main">
</div>
<a href='javascript:creatDiv();'>創建元素</a>
[br]
<a href='javascript:removeDiv();'>刪除元素</a>
[br]
<a href='javascript:checkDiv();'>測試DIV是否還存在</a>
</body>
</html>
[Ctrl+A 全選 注:如需引入外部Js需刷新才能執行]
點擊“創建元素”後再點擊“刪除元素”將新創建的元素用 removeChild 將其刪除,再點擊“測試DIV是否還存在”來查看元素是否真的刪除了,結果 alert 顯示
null 。看來元素結點真的已經被刪除了。那司徒正美文中所說到的內存洩露又是怎麼一種情況呢?只好上 google 搜索,看是否有人也遇到 removeChild 引起內
存洩漏的問題。終於在一英文版的 msdn 發現有人在問同樣的問題(LINK),我將它裡面的代碼稍微修改一下通過對比的方式來看一下 removeChild 引起內存洩漏的情況。
代碼如下:
復制代碼 代碼如下:
<html>
<head>
<title>測試 removeChild 導致的內存洩漏</title>
</head>
<body>
<a href="javascript:leak();">產生內存洩漏方式</a>
<br />
<a href="javascript:notLeak();">不產生內存洩漏方式</a>
</body>
</html>
<script>
var dialog;
function add()
{
dialog = document.createElement('div');
var html = '<div><p>Title</p></div>';
dialog.innerHTML = html;
document.body.appendChild(dialog);
dialog.style.marginTop='200px';
dialog.style.marginLeft='200px';
}
function remove()
{
document.body.removeChild(dialog);
dialog=null;
}
function leak()
{
for(var i=0;i<100000;i++){
add();
remove();
}
alert('leak done');
}
function notLeak()
{
for(var i=0;i<100000;i++){
add();
discardElement(dialog);
}
alert('notLeak done');
}
function discardElement(element) {
var garbageBin = document.getElementById('IELeakGarbageBin');
if (!garbageBin) {
garbageBin = document.createElement('DIV');
garbageBin.id = 'IELeakGarbageBin';
garbageBin.style.display = 'none';
document.body.appendChild(garbageBin);
}
// move the element to the garbage bin
garbageBin.appendChild(element);
garbageBin.innerHTML = '';
}
</script>
[Ctrl+A 全選 注:如需引入外部Js需刷新才能執行]
首先運行“產生內存洩漏方式”
未運行前打開任務管理器監控內存大小如下:
運行完再查看內存大小,可以看到內存大小已經增加了很多。
接著我再運行“不產生內存洩漏方式”
未運行前打開任務管理器監控內存大小如下:
運行完再查看內存大小,可以看到內存較“產生內存洩漏方式”小了很多。
PS: 為了檢驗 removeChild 導致的內存洩漏 ,我 google 了很多 IE 內存洩漏的相關文章。
相關文章如下:
http://www.cnblogs.com/dwjaissk/archive/2007/07/20/824884.html
http://bugs.dojotoolkit.org/ticket/1727
http://article.yeeyan.org/view/3407/10103