日本综合一区二区|亚洲中文天堂综合|日韩欧美自拍一区|男女精品天堂一区|欧美自拍第6页亚洲成人精品一区|亚洲黄色天堂一区二区成人|超碰91偷拍第一页|日韩av夜夜嗨中文字幕|久久蜜综合视频官网|精美人妻一区二区三区

RELATEED CONSULTING
相關咨詢
選擇下列產(chǎn)品馬上在線溝通
服務時間:8:30-17:00
你可能遇到了下面的問題
關閉右側工具欄

新聞中心

這里有您想知道的互聯(lián)網(wǎng)營銷解決方案
JS魔法堂:再識IE的內(nèi)存泄露

一、前言                            

公司主營業(yè)務:網(wǎng)站設計、網(wǎng)站建設、移動網(wǎng)站開發(fā)等業(yè)務。幫助企業(yè)客戶真正實現(xiàn)互聯(lián)網(wǎng)宣傳,提高企業(yè)的競爭能力。創(chuàng)新互聯(lián)是一支青春激揚、勤奮敬業(yè)、活力青春激揚、勤奮敬業(yè)、活力澎湃、和諧高效的團隊。公司秉承以“開放、自由、嚴謹、自律”為核心的企業(yè)文化,感謝他們對我們的高要求,感謝他們從不同領域給我們帶來的挑戰(zhàn),讓我們激情的團隊有機會用頭腦與智慧不斷的給客戶帶來驚喜。創(chuàng)新互聯(lián)推出桓仁免費做網(wǎng)站回饋大家。

IE6~8除了不遵守W3C標準和各種詭異外,我想最讓人詬病的應該是內(nèi)存泄露的問題了。這陣子趁項目技術調(diào)研的機會好好的再認識一回,以下內(nèi)容若有紕漏請大家指正,謝謝!

目錄一大坨!

二、內(nèi)存泄漏到底是哪里漏了?

2.1. JS Engine Object、DOM Element 和 BOM Element

2.2. JS Engine Object的內(nèi)存回收機制

2.3. DOM Element的內(nèi)存回收機制

2.4. 兩種泄漏方式

三、4種泄漏模式

3.1. Circular References

3.2. Closures

3.3. Cross-page Leaks

3.4. Pseduo-Leaks

四、當前頁面泄漏的示例

4.1. DOM Hyperspace引起的DOM Element引用孤島

4.2. 釋放Iframe沒那么簡單

五、IE8下連續(xù)修改IMG的src居然耗盡內(nèi)存?

六、監(jiān)控工具

七、總結

八、參考

#p#

二、內(nèi)存泄漏到底是哪里漏了?                  

SPA跑久了頁面響應速度劇減又被用戶投訴,搪塞說句“IE是比較容易發(fā)生內(nèi)存泄漏,刷刷頁面就好”。那真的是刷刷頁面就能釋放泄漏了的內(nèi)存嗎?下面我們一起來探討一下!

內(nèi)存泄漏:內(nèi)存資源得不到釋放 && 失去對該內(nèi)存區(qū)的指針 => 無法復用內(nèi)存資源,最終導致內(nèi)存溢出

2.1. JS Engine Object、DOM Element 和 BOM Element

Script中我們能操作的對象可分為三種:JS Engine Object、DOM Element 和 BOM Element。

JS Engine Object: var obj = Object(); var array = [];等等

DOM Element: var el = document.createElement('div'); var div = document.getElementById('name');等等

BOM Element: window; window.location;等等

其中只有JS Engine Object和DOM Element是我們可以CRUD的,因此也就有可能發(fā)生內(nèi)存泄漏的問題。

2.2. JS Engine Object的內(nèi)存回收機制 

IE的JScript Garbage Collector采用的是Mark-and-Sweep算法,當執(zhí)行垃圾回收時會先遍歷所有JS Engine Object并標記未被引用的對象,然后釋放掉被標記的內(nèi)存空間。

由于Mark-and-Sweep算法的緣故,也能很好地釋放引用孤島的內(nèi)存空間。

而IE下獨有的CollectGarbage()則用于回收無引用或引用孤島的JS Engine Object。

2.3. DOM Element的內(nèi)存回收機制

當DOM Element不再被引用時會被回收,但具體被誰何時回收則有待研究了。

2.4. 兩種泄漏方式

a. 當前頁面泄漏:刷新頁面或跳轉到其他頁面就能釋放的內(nèi)存資源。

b. 跨頁面泄漏:刷新頁面或跳轉到其他頁面也無法釋放的內(nèi)存資源。

當前頁面泄漏處理難度相對簡單,跨頁面泄漏才是處理大頭。

#p#

三、4種泄漏模式

下面是Justin Rogers總結出來的4種會引起泄漏的反模式。

3.1. Circular References(導致跨頁面內(nèi)存泄漏)

循環(huán)引用可謂是引起內(nèi)存泄漏的根本原因,其他的泄漏模式***層還是因為出現(xiàn)的循環(huán)引用。

Leak Memory

 
 
 
  •  
  •  Non-Leak Memory

     
     
    1.  
     
  •  
  •  
  • 3.2. Closures(導致跨頁面內(nèi)存泄漏)

    閉包具有Lexical scope特性,延長了方法參數(shù)和局部變量的生命周期,但同時又容易在無意當中引入循環(huán)引用的問題。

    Leak Memory

     
     
     
  •  
  •  Non-Leak Memory

     
     
     
  •  
  •   3.3. Cross-page Leaks(當前頁面內(nèi)存泄漏)

    由于節(jié)點建立聯(lián)系時會尋找scope,若沒有則創(chuàng)建temporary scope,若有則拋棄原有的temporary scope采用已有的scope。

    Leak Memory

     
     
    1.  
    2.  
    3.  
    4.  
    5.  
    6.  Memory Leaking Insert  
     
  •  
  •  
  •  當childDiv與parentDiv建立連接時,為讓childDiv能獲取parentDiv的信息,IE會創(chuàng)建temporary scope。而當將parentDiv添加到DOM tree中時,則childDiv和parentDiv均繼承document的scope,而temporary scope卻不會被GC釋放,而要等待瀏覽器刷新頁面才能清理。

    Non-Leak Memory

     
     
    1.  
    2.  
    3.  
    4.  
    5.  
    6.  Clean Insert  
     
  •  
  •  
  • 一直使用document scope,不會創(chuàng)建temporary scope

    3.4. Pseduo-Leaks

    連續(xù)創(chuàng)建多個JS Engine Object,而GC未能及時釋放內(nèi)存,其實根本就不是內(nèi)存泄漏

    var tmpStr
    for(var i = 0; i < 100000; ++i)
    tmpStr = "test"

    #p#

    四、當前頁面泄漏的示例                      

    4.1. DOM Hyperspace引起的DOM Element引用孤島

    DOM Hyperspace由PPK發(fā)現(xiàn),在IE下通過removeChild或removeNode從父節(jié)點(無論是否已加入DOM Tree)中移除節(jié)點后,會創(chuàng)建一個新的#documentFragment,并且被移除的節(jié)點的parentNode為 該#documentFragment,而該#documentFragment.firstChild為被移除的節(jié)點,因此存在DOM Element間的circular reference導致無法釋放,只有刷新頁面后才會釋放資源。

    Leak Memory

    var div = document.createElement('div')
    document.body.appendChild(div)
    div.parentNode.removeChild(div)

    alert(div.parentNode) // IE8下為[Object object],Chrome等瀏覽器為null

    Non-Leak Memory

     
     
    1. function rm(el){ 
    2. if (!+'\v1'){ 
    3. var d = document.createElement('div') 
    4. d.appendChild(el) 
    5. d.innerHTML = '' 
    6. else{ 
    7. el.parentNode.removeChild(el) 
    8.  
    9. var div = document.createElement('div') 
    10. document.body.appendChild(div) 
    11. rm(div) 
    12.  
    13. alert(div.parentNode) // IE8下為null 

    4.2. 釋放Iframe沒那么簡單

    iframe所占的資源有兩部分:iframe元素所占的內(nèi)存空間 和 iframe內(nèi)頁面所占的內(nèi)存空間。

    內(nèi)存空間釋放步驟:

      1. 釋放 iframe內(nèi)頁面所占的內(nèi)存空間

        通過設置src=''或src='about:blank'來釋放內(nèi)部頁面的資源

      2. 釋放 iframe元素所占的內(nèi)存空間

        通過removeChild、removeNode等方法釋放iframe元素的內(nèi)存空間

    ligerTab1.2.1的清除方式

    var iframe = ...
    iframe.src = 'about:blank'
    iframe.contentWindow.document.write('')
    CollectGarbage && CollectGarbage()
    iframe.parentNode.removeChild(iframe)

    #p#

    五、IE8下連續(xù)修改IMG的src居然耗盡內(nèi)存?            

    由于IE8會對非原始尺寸的圖片進行抗鋸齒平滑處理,從而消耗更多的CPU和內(nèi)存資源。當圖片大小和尺寸到一定時,則會出現(xiàn)掛死的情況。(IE6、7沒有抗鋸齒平滑處理,而IE9則移除該功能)

    而這種情況當然就不屬于Memory Leak啦!

    題外話:

    眾所周知IMG是replaced element,其width和height屬性缺省值又外部資源決定,而我們通過CSS設置的width和height屬性均是對缺省值的二次加工。

    假設圖片原始尺寸為width:200px/height:400px,現(xiàn)在通過CSS設置width:100px,那么圖片將按等比例縮放為 width:100px/height:200px;但通過CSS設置width:100px/height:100px時,那么圖片則不是按等比例縮放 了。

    #p#

    六、監(jiān)控工具                           

    監(jiān)控方式多種多樣,這里大概分為兩類:

    1. 當前頁面泄漏:Windows的任務管理器、Chrome->dev tools->Profiles->Take Heap Snapshot/Record Heap Allocations等等

    2. 跨頁面泄漏:sIEve

    操作步驟:

    1. 在Address輸入框輸入網(wǎng)址,點擊Go (瀏覽網(wǎng)頁)

    2. 執(zhí)行測試用例

    3. 點擊about:blank按鈕(跳轉到空白頁)

    4. 查看#leaks列下是否有增長,有則表示出現(xiàn)跨頁面的內(nèi)存泄漏

    七、總結                            

    上述內(nèi)容以概念為主,最終還是要實戰(zhàn)來驗證和完善、補充。

    來自:肥子John^_^ http://www.cnblogs.com/fsjohnhuang/p/4455822.htm

    八、參考                            

     What are closures?

      Understanding and Solving Internet Explorer Leak Patterns

      JavaScript and memory leaks


    當前名稱:JS魔法堂:再識IE的內(nèi)存泄露
    鏈接URL:http://www.dlmjj.cn/article/cocjcde.html