前端性能優(yōu)化(總結(jié))

(1)減少HTTP請求次數(shù)
  • 盡量合并圖片碉熄、CSS籽懦、JS暴拄。
舉例:
如有5個css文件漓滔,則需要發(fā)出5次http請求,第一次訪問頁面需要等待很長世間乖篷。
如合并成1個css文件响驴,則發(fā)出1次http請求,節(jié)省網(wǎng)絡(luò)請求時間撕蔼,加快頁面的加載豁鲤。
  • Expires和Cache-Control (多由服務(wù)器端去設(shè)置)
    瀏覽器在第一次訪問頁面時向服務(wù)器請求資源,并緩存起來鲸沮,下次再訪問時會判斷在緩存中是否已有該資源且有沒有更新過琳骡,如果已有該資源且未更新過,則直接從瀏覽器緩存中讀取讼溺。
(1)Expires 表示存在時間:
允許客戶端在這個時間之前不去檢查(發(fā)請求)楣号,等同max-age的效果。但是如果同時存在
則被Cache-Control的max-age覆蓋怒坯。
(2)Cache-control用于控制HTTP緩存:
1.原理
通過HTTP的Expires和Cache-Control炫狱,使已緩存資源不再發(fā)起http請求。
(將靜態(tài)內(nèi)容設(shè)為永不過期剔猿,或者很長時間后才過期视译。)
2.應(yīng)用
通過HTTP的META設(shè)置expires和cache-control
<meta http-equiv="Cache-Control" content="max-age=7200" />
<meta http-equiv="Expires" content="Mon, 20 Jul 2009 23:00:00 GMT" />
上述設(shè)置僅為舉例,實際使用其一即可艳馒。
這樣寫的話僅對該網(wǎng)頁有效憎亚,對網(wǎng)頁中的圖片或其他請求無效员寇,并不會做任cache。
(2)對HTTP傳輸進行壓縮 (gzip壓縮)(多由服務(wù)端去設(shè)置)

即在js第美,css蝶锋、圖片等資源已經(jīng)壓縮的基礎(chǔ)上,在HTTP傳輸過程中的再次壓縮什往。

1.原理
客戶端:通過Accept-Encoding頭來聲明瀏覽器支持的壓縮方式扳缕,
服務(wù)端:通過content-Encoding來啟用壓縮,配置壓縮的文件類型别威,壓縮方式躯舔。
(當(dāng)客戶端的請求到達服務(wù)器,服務(wù)器對資源進行壓縮后省古,返回給客戶端粥庄,客戶端按照相應(yīng)的方式進行解析。)

客戶端(HTTP請求頭)——accept-encoding: gzip, deflate, sdch, br
服務(wù)器(HTTP響應(yīng)頭)——content-encoding:gzip
2.應(yīng)用
以tomcat服務(wù)器的配置為例:
找到tomcat安裝目錄下的conf文件夾下的server.xml文件豺妓,進行如下配置惜互,重啟tomcat即可:
① compress="on" :表示開啟壓縮。
② compressionMinSize="2048":表示對大于2KB的文件進行壓縮
③ compressableMimeType="text/html,text/xml,application/
JavaScript,text/css,text/plain,image/png,image/jpeg,image/gif" //表示將進行壓縮的文件類型

注意:不應(yīng)該對圖片進行再壓縮琳拭,因為圖片本身已經(jīng)被壓縮過
(如果再進行g(shù)zip壓縮训堆,可能得到的結(jié)果是和圖片本身大小相差不大,純粹是浪費服務(wù)器的CPU資源來做無用功白嘁。)

3.優(yōu)缺點
對HTTP傳輸內(nèi)容進行壓縮的優(yōu)坑鱼、缺點:
① 優(yōu)點:減少HTTP響應(yīng)時間,提升傳輸效率絮缅。
② 缺點:壓縮過程占用服務(wù)器額外的CPU周期鲁沥,客戶端也要對壓縮文件進行解壓縮,這也需要占用部分時間盟蚣。
(3)將CSS文件放在頂部
原理:
將CSS放在底部黍析,頁面可以逐步呈現(xiàn),
但在CSS下載并解析完畢后屎开,已經(jīng)呈現(xiàn)的文字和圖片就要需要根據(jù)新的樣式重繪阐枣,這是一種不好的用戶體驗。
(4)將javascript放在底部
原理:
script沒有async和defer時奄抽,JS文件將在下載后立即執(zhí)行蔼两。這種情況下,script放在頂部會阻塞頁面呈現(xiàn)逞度,
在網(wǎng)速慢的情況下會導(dǎo)致“白屏”额划,直到腳本下載完畢才繼續(xù)呈現(xiàn)頁面。因此档泽,script放在底部可以讓頁面盡快呈現(xiàn)俊戳。

script全部放在head中會出現(xiàn)的問題:
在需要操作body中的某元素時揖赴,可能找不到該元素,因此抑胎,若要放在head中燥滑,
一般需要綁定一個監(jiān)聽windows.onload=function(){ ... },
當(dāng)文檔全部解析完之后再執(zhí)行script代碼阿逃。
(5)cookie優(yōu)化
cookie原理:

1铭拧、去除沒有必要的cookie,如果網(wǎng)頁不需要cookie就完全禁掉恃锉。

2搀菩、將cookie的大小減到最小。
由于cookie在訪問對應(yīng)域名下的資源時都會通過HTTP請求發(fā)送到服務(wù)器破托,
因此肪跋,減小cookie的大小,能減小HTTP請求報文的大小炼团,提高響應(yīng)速度澎嚣。

3、設(shè)置合適的過期時間瘟芝,較長的過期時間可以提高響應(yīng)速度。
給cookie添加一個過期時間褥琐,則cookie信息將存儲到硬盤上锌俱,即使瀏覽器退出Cookie還會存
在。只要Cookie未被清除且還在過期時間內(nèi)敌呈,該Cookie就會在訪問對應(yīng)域名時發(fā)送給服務(wù)器贸宏。

4、通過使用不同的domain減少cookie的使用磕洪。
cookie在訪問對應(yīng)域名下的資源時都會通過HTTP請求發(fā)送到服務(wù)器吭练,但在訪問一些資源,如js析显,css和圖片時鲫咽,
大多數(shù)情況下cookie是多余的,可以使用不同的domain來存儲這些靜態(tài)資源谷异,這樣訪問這些資源時就不會發(fā)送
多余的cookie分尸,從而提高響應(yīng)速度。
(6)可緩存的AJAX
異步請求同樣的造成用戶等待歹嘹,所以使用ajax請求時箩绍,要主動告訴瀏覽器如果該請求有緩存就去請求緩存內(nèi)容。
如下代碼片段: cache:true就是顯式的要求如果當(dāng)前請求有緩存的話尺上,直接使用緩存
$.ajax({
     url : 'url',
     dataType : "json",
     cache: true,   //如果有緩存材蛛,就直接使用緩存
     success : function(son, status){    
      }
(7)使用GET來完成AJAX請求
當(dāng)使用XMLHttpRequest時圆到,瀏覽器中的POST方法是一個“兩步走”的過程:首先發(fā)送文件頭,然后才發(fā)送數(shù)據(jù)卑吭。

因此使用GET獲取數(shù)據(jù)時更加有意義构资。
(8) 避免404
比如外鏈的css或者js文件出現(xiàn)問題返回404時,會破壞瀏覽器對文件的并行加載陨簇。

并且瀏覽器會把試圖在返回的404響應(yīng)內(nèi)容中找到可能有用的部分當(dāng)作JavaScript代碼來執(zhí)行吐绵。
(9)避免使用CSS表達式
例如:
font-color: expression( (new Date()).getHours()%3 ? “#FFFFFF" : “#AAAAAA" );

這個表達式會持續(xù)的在頁面上計算樣式,影響頁面的性能河绽。并且css表達式只被IE支持己单。


(10)避免空的src和href
當(dāng)link標(biāo)簽的href屬性為空、script標(biāo)簽的src屬性為空的時候耙饰,瀏覽器渲染的時候會把當(dāng)前頁面的URL

作為它們的屬性值纹笼,從而把頁面的內(nèi)容加載進來作為它們的值。所以要避免犯這樣的疏忽
(11)縮小favicon.ico并緩存

https://www.ico.la/ 提供的在線免費創(chuàng)建favicon.ico文件服務(wù).

(1)使它盡量在1KB左右苟跪。使用ico格式廷痘,不要使用png,jpg等其他格式。
(2)將該文件放在單獨的主機中,例如 images.mydomain.com . 這樣可以避免在請求該文件時發(fā)送cookie. 
(3)緩存
(12)不要在HTML中縮放圖片
需要的圖片尺寸是50* 50件已,不要用500*500的圖片在img中設(shè)置成50*50
<img width=”50″ height=”50″ src=“hahah.jpg” alt=”hahaha” />
(13)使用CDN
網(wǎng)站上靜態(tài)資源即css笋额、js全都使用cdn分發(fā),圖片亦然篷扩。
(14)減少dom元素數(shù)量
原理:
減少DOM數(shù)量兄猩,就會減少瀏覽器的解析負(fù)擔(dān)
(15)不要使用濾鏡
IE獨有屬性AlphaImageLoader用于修正7.0以下版本中顯示PNG圖片的半透明效果。
這個濾鏡的問題在于瀏覽器加載圖片時它會終止內(nèi)容的呈現(xiàn)并且凍結(jié)瀏覽器鉴未。在每一個元素(不僅僅是圖片)
它都會運算一次枢冤,增加了內(nèi)存開支,因此它的問題是多方面的铜秆。完全避免使用AlphaImageLoader的最好方法
就是使用PNG8格式來代替淹真,這種格式能在IE中很好地工作。如果你確實需要使用AlphaImageLoader连茧,
請使用下劃線_filter又使之對IE7以上版本的用戶無效核蘸。



Javascript優(yōu)化篇

  • 避免全局查找
    在一個函數(shù)中會用到全局對象存儲為局部變量來減少全局查找,因為訪問局部變量的速度要比訪問全局變量的速度更快些
  function search() {
            //當(dāng)我要使用當(dāng)前頁面地址和主機域名
            alert(window.location.href + window.location.host);
        }
        //最好的方式是如下這樣  先用一個簡單變量保存起來
        function search() {
            var location = window.location;
            alert(location.href + location.host);
        }
  • 定時器
    如果針對的是不斷運行的代碼梅屉,不應(yīng)該使用setTimeout值纱,而應(yīng)該是用setInterval,因為setTimeout每一次都會初始化一個定時器坯汤,而setInterval只會在開始的時候初始化一個定時器
var timeoutTimes = 0;
        function timeout() {
            timeoutTimes++;
            if (timeoutTimes < 10) {
                setTimeout(timeout, 10);
            }
        }
        timeout();
        //可以替換為:
        var intervalTimes = 0;
        function interval() {
            intervalTimes++;
            if (intervalTimes >= 10) {
                clearInterval(interv);
            }
        }
        var interv = setInterval(interval, 10);
  • 字符串連接
    如果要連接多個字符串虐唠,應(yīng)該少使用+=,如
    s+=a;
    s+=b;
    s+=c;
    應(yīng)該寫成s+=a + b + c惰聂;
    而如果是收集字符串疆偿,比如多次對同一個字符串進行+=操作的話咱筛,最好使用一個緩存,使用JavaScript數(shù)組來收集杆故,最后使用join方法連接起來
     var buf = [];
        for (var i = 0; i < 100; i++) {
            buf.push(i.toString());
        }
        var all = buf.join("");
  • 避免with語句
    和函數(shù)類似 迅箩,with語句會創(chuàng)建自己的作用域,因此會增加其中執(zhí)行的代碼的作用域鏈的長度处铛,由于額外的作用域鏈的查找饲趋,在with語句中執(zhí)行的代碼肯定會比外面執(zhí)行的代碼要慢,在能不使用with語句的時候盡量不要使用with語句撤蟆。
with (a.b.c.d) {
            property1 = 1;
            property2 = 2;
        }
        //可以替換為:
        var obj = a.b.c.d;
        obj.property1 = 1;
        obj.property2 = 2;
  • 數(shù)字轉(zhuǎn)換成字符串
    般最好用”" + 1來將數(shù)字轉(zhuǎn)換成字符串奕塑,雖然看起來比較丑一點,但事實上這個效率是最高的家肯,性能上來說:
    (“” +) > String() > .toString() > new String()

  • 重復(fù)使用的調(diào)用結(jié)果龄砰,事先保存到局部變量

 //避免多次取值的調(diào)用開銷
        var h1 = element1.clientHeight + num1;
        var h2 = element1.clientHeight + num2;
        //可以替換為:
        var eleHeight = element1.clientHeight;
        var h1 = eleHeight + num1;
        var h2 = eleHeight + num2;
  • 條件分支
    (1)將條件分支,按可能性順序從高到低排列:可以減少解釋器對條件的探測次數(shù)
    (2)在同一條件子的多(>2)條件分支時讨衣,使用switch優(yōu)于if:switch分支選擇的效率高于if换棚,在IE下尤為明顯。4分支的測試反镇,IE下switch的執(zhí)行時間約為if的一半固蚤。
    (3)使用三目運算符替代條件分支
 if (a > b) {
            num = a;
        } else {
            num = b;
        }
        //可以替換為:
        num = a > b ? a : b;
  • 避免雙重解釋
    如果要提高代碼性能,盡可能避免出現(xiàn)需要按照J(rèn)avaScript解釋的字符串愿险,也就是
    (1)盡量少使用eval函數(shù)
    使用eval相當(dāng)于在運行時再次調(diào)用解釋引擎對內(nèi)容進行運行颇蜡,需要消耗大量時間,而且使用Eval帶來的安全性問題也是不容忽視的辆亏。
    (2)不要使用Function構(gòu)造器
    不要給setTimeout或者setInterval傳遞字符串參數(shù)
    var num = 0;
        setTimeout('num++', 10);
        //可以替換為:
        var num = 0;
        function addNum() {
            num++;
        }
        setTimeout(addNum, 10);
  • 使用一次innerHTML賦值代替構(gòu)建dom元素
    對于大的DOM更改,使用innerHTML要比使用標(biāo)準(zhǔn)的DOM方法創(chuàng)建同樣的DOM結(jié)構(gòu)快得多鳖目。
   var frag = document.createDocumentFragment();
        for (var i = 0; i < 1000; i++) {
            var el = document.createElement('p');
            el.innerHTML = i;
            frag.appendChild(el);
        }
        document.body.appendChild(frag);
        //可以替換為:
        var html = [];
        for (var i = 0; i < 1000; i++) {
            html.push('<p>' + i + '</p>');
        }
        document.body.innerHTML = html.join('');
  • 使用DocumentFragment優(yōu)化多次append
    一旦需要更新DOM,請考慮使用文檔碎片來構(gòu)建DOM結(jié)構(gòu)扮叨,然后再將其添加到現(xiàn)存的文檔中。
for (var i = 0; i < 1000; i++) {
            var el = document.createElement('p');
            el.innerHTML = i;
            document.body.appendChild(el);
        }
        //可以替換為:
        var frag = document.createDocumentFragment();
        for (var i = 0; i < 1000; i++) {
            var el = document.createElement('p');
            el.innerHTML = i;
            frag.appendChild(el);
        }
        document.body.appendChild(frag);
  • 浮點數(shù)轉(zhuǎn)換成整型
    很多人喜歡使用parseInt()领迈,其實parseInt()是用于將字符串轉(zhuǎn)換成數(shù)字彻磁,而不是浮點數(shù)和整型之間的轉(zhuǎn)換,我們應(yīng)該使用Math.floor()或者Math.round()

  • 使用firstChild和nextSibling代替childNodes遍歷dom元素

  • ==和===的區(qū)別
    避免在if和while語句的條件部分進行賦值狸捅,如if (a = b)衷蜓,應(yīng)該寫成if (a == b),但是在比較是否相等的情況下尘喝,最好使用全等運行符磁浇,也就是使用===和!==操作符會相對于==和!=會好點。==和!=操作符會進行類型強制轉(zhuǎn)換

 var valueA = "1";
        var valueB = 1;
        if (valueA == valueB) {
            alert("Equal");
        }
        else {
            alert("Not equal");
        }
        //output: "Equal"
        if (valueA === valueB) {
            alert("Equal");
        }
        else {
            alert("Not equal");
        }
        //output: "Not equal"
  • 何時用單引號朽褪,何時用雙引號
    雖然在JavaScript當(dāng)中置吓,雙引號和單引號都可以表示字符串, 為了避免混亂无虚,我們建議在HTML中使用雙引號,在JavaScript中使用單引號衍锚,但為了兼容各個瀏覽器友题,也為了解析時不會出錯,定義JSON對象時戴质,最好使用雙引號

  • 使用return語句需要注意
    一條有返回值的return語句不要用()括號來括住返回值度宦,如果返回表達式,則表達式應(yīng)與return關(guān)鍵字在同一行告匠,以避免壓縮時戈抄,壓縮工具自動加分號而造成返回與開發(fā)人員不一致的結(jié)果

   function F1() {
            var valueA = 1;
            var valueB = 2;
            return valueA + valueB;
        }
        function F2() {
            var valueA = 1;
            var valueB = 2;
            return
            valueA + valueB;
        }
        alert(F1());  //output: 3 
        alert(F2());  //ouput: undefined
  • 每條語句末尾須加分號
    在if語句中,即使條件表達式只有一條語句也要用{}把它括起來凫海,以免后續(xù)如果添加了語句之后造成邏輯錯

  • 使用+號時需謹(jǐn)慎
    JavaScript 和其他編程語言不同的是呛凶,在 JavaScript 中,’+'除了表示數(shù)字值相加行贪,字符串相連接以外漾稀,還可以作一元運算符用,把字符串轉(zhuǎn)換為數(shù)字建瘫。因而如果使用不當(dāng)崭捍,則可能與自增符’++’混淆而引起計算錯誤

 var valueA = 20;
        var valueB = "10";
        alert(valueA + valueB);     //ouput: 2010 
        alert(valueA + (+valueB));  //output: 30 
        alert(valueA + +valueB);    //output:30 
        alert(valueA ++ valueB);     //Compile error
  • switch語句相對if較快
    通過將case語句按照最可能到最不可能的順序進行組織

  • 位運算較快
    當(dāng)進行數(shù)字運算時,位運算操作要比任何布爾運算或者算數(shù)運算快

  • 解耦CSS/JavaScript
    顯示問題的唯一來源應(yīng)該是CSS啰脚,行為問題的唯一來源應(yīng)該是JavaScript殷蛇,層次之間保持松散耦合才可以讓你的應(yīng)用程序更加易于維護,所以像以下的代碼element.style.color=”red”盡量改為element.className=”edit”橄浓,而且不要在css中通過表達式嵌入JavaScript

  • 解耦HTML/JavaScript
    JavaScript和HTML的緊密耦合:直接寫在HTML中的JavaScript粒梦、使用包含內(nèi)聯(lián)代碼的<script>元素、使用HTML屬性來分配事件處理程序等
    HTML和JavaScript的緊密耦合:JavaScript中包含HTML荸实,然后使用innerHTML來插入一段html文本到頁面
    其實應(yīng)該是保持層次的分離匀们,這樣可以很容易的確定錯誤的來源,所以我們應(yīng)確保HTML呈現(xiàn)應(yīng)該盡可能與JavaScript保持分離

  • 解耦應(yīng)用程序/事件處理程序
    將應(yīng)用邏輯和事件處理程序相分離:一個事件處理程序應(yīng)該從事件對象中提取准给,并將這些信息傳送給處理應(yīng)用邏輯的某個方法中泄朴。這樣做的好處首先可以讓你更容易更改觸發(fā)特定過程的事件,其次可以在不附加事件的情況下測試代碼露氮,使其更易創(chuàng)建單元測試

  • 釋放javascript對象
    在rich應(yīng)用中祖灰,隨著實例化對象數(shù)量的增加,內(nèi)存消耗會越來越大畔规。所以應(yīng)當(dāng)及時釋放對對象的引用局扶,讓GC能夠回收這些內(nèi)存控件。
    對象:obj = null
    對象屬性:delete obj.myproperty
    數(shù)組item:使用數(shù)組的splice方法釋放數(shù)組中不用的item

  • 釋放dom元素占用的內(nèi)存
    將dom元素的innerHTML設(shè)置為空字符串,可以釋放其子元素占用的內(nèi)存详民。
    在rich應(yīng)用中延欠,用戶也許會在一個頁面上停留很長時間,可以使用該方法釋放積累得越來越多的dom元素使用的內(nèi)存沈跨。

  • 避免string的隱式裝箱
    對string的方法調(diào)用由捎,比如’xxx’.length,瀏覽器會進行一個隱式的裝箱操作饿凛,將字符串先轉(zhuǎn)換成一個String對象狞玛。推薦對聲明有可能使用String實例方法的字符串時,采用如下寫法:
    var myString = new String(‘Hello World’);

  • 通過javascript創(chuàng)建的dom對象涧窒,必須append到頁面中
    IE下心肪,腳本創(chuàng)建的dom對象,如果沒有append到頁面中纠吴,刷新頁面硬鞍,這部分內(nèi)存是不會回收的!

       function create() {
            var gc = document.getElementById('GC');
            for (var i = 0; i < 5000; i++) {
                var el = document.createElement('div');
                el.innerHTML = "test";
                //下面這句可以注釋掉戴已,看看瀏覽器在任務(wù)管理器中固该,點擊按鈕然后刷新后的內(nèi)存變化
                gc.appendChild(el);
            }
        }
  • 避免全局量(使用閉包函數(shù))
    全局變量應(yīng)該全部字母大寫,各單詞之間用_下劃線來連接糖儡。盡可能避免全局變量和函數(shù), 盡量減少全局變量的使用伐坏,因為在一個頁面中包含的所有JavaScript都在同一個域中運行。所以如果你的代碼中聲明了全局變量或者全局函數(shù)的話握联,后面的代碼中載入的腳本文件中的同名變量和函數(shù)會覆蓋掉(overwrite)你的桦沉。
//糟糕的全局變量和全局函數(shù)
var current = null;
function init(){
//...
}
function change() {
    //...
}
function verify() {
    //...
}
--
--
//解決辦法有很多,Christian Heilmann建議的方法是:
//如果變量和函數(shù)不需要在“外面”引用金闽,那么就可以使用一個沒有名字的方法將他們?nèi)及饋怼?閉包)
(function(){
var current = null;
function init() {
    //...
}
function change() {
    //...
}
function verify() {
    //...
}
})();
--
--
//如果變量和函數(shù)需要在“外面”引用纯露,需要把你的變量和函數(shù)放在一個“命名空間”中
//我們這里用一個function做命名空間而不是一個var,因為在前者中聲明function更簡單代芜,而且能保護隱私數(shù)據(jù)
myNameSpace = function() {
    var current = null;

    function init() {
        //...
    }
    function change() {
        //...
    }
    function verify() {
        //...
    }
//所有需要在命名空間外調(diào)用的函數(shù)和屬性都要寫在return里面
    return {
        init: init,
        //甚至你可以為函數(shù)和屬性命名一個別名
        set: change
    };
};
  • 尊重對象的所有權(quán)
    (1)不要為實例或原型添加屬性
    (2)不要為實例或者原型添加方法
    (3)不要重定義已經(jīng)存在的方法
    (4)不要重復(fù)定義其它團隊成員已經(jīng)實現(xiàn)的方法苔埋,永遠不要修改不是由你所有的對象,你可以通過以下方式為對象創(chuàng)建新的功能:
    (5)創(chuàng)建包含所需功能的新對象蜒犯,并用它與相關(guān)對象進行交互
    (6)創(chuàng)建自定義類型,繼承需要進行修改的類型荞膘,然后可以為自定義類型添加額外功能

-循環(huán)引用
如果循環(huán)引用中包含DOM對象或者ActiveX對象罚随,那么就會發(fā)生內(nèi)存泄露。內(nèi)存泄露的后果是在瀏覽器關(guān)閉前羽资,即使是刷新頁面淘菩,這部分內(nèi)存不會被瀏覽器釋放。

簡單的循環(huán)引用: var el = document.getElementById('MyElement');
        var func = function () {
            //…
        }
        el.func = func;
        func.element = el;
但是通常不會出現(xiàn)這種情況。通常循環(huán)引用發(fā)生在為dom元素添加閉包作為expendo的時候潮改。
function init() {
            var el = document.getElementById('MyElement');
            el.onclick = function () {
                //……
            }
        }
        init();
init在執(zhí)行的時候狭郑,當(dāng)前上下文我們叫做context。
這個時候汇在,context引用了el翰萨,el引用了function,function引用了context糕殉。這時候形成了一個循環(huán)引用亩鬼。
//解決方如下
//解決方如下
//解決方如下
1)  置空dom對象
 function init() {
            var el = document.getElementById('MyElement');
            el.onclick = function () {
                //……
            }
        }
        init();
        //可以替換為:
        function init() {
            var el = document.getElementById('MyElement');
            el.onclick = function () {
                //……
            }
            el = null;
        }
        init();
將el置空,context中不包含對dom對象的引用阿蝶,從而打斷循環(huán)應(yīng)用雳锋。
如果我們需要將dom對象返回,可以用如下方法:
  function init() {
            var el = document.getElementById('MyElement');
            el.onclick = function () {
                //……
            }
            return el;
        }
        init();
        //可以替換為:
        function init() {
            var el = document.getElementById('MyElement');
            el.onclick = function () {
                //……
            }
            try {
                return el;
            } finally {
                el = null;
            }
        }
        init();
//解決方如下
//解決方如下
//解決方如下
2)  構(gòu)造新的context
   function init() {
            var el = document.getElementById('MyElement');
            el.onclick = function () {
                //……
            }
        }
        init();
        //可以替換為:
        function elClickHandler() {
            //……
        }
        function init() {
            var el = document.getElementById('MyElement');
            el.onclick = elClickHandler;
        }
        init();
把function抽到新的context中羡洁,這樣玷过,function的context就不包含對el的引用,從而打斷循環(huán)引用筑煮。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末辛蚊,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子咆瘟,更是在濱河造成了極大的恐慌嚼隘,老刑警劉巖,帶你破解...
    沈念sama閱讀 211,123評論 6 490
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件袒餐,死亡現(xiàn)場離奇詭異飞蛹,居然都是意外死亡,警方通過查閱死者的電腦和手機灸眼,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,031評論 2 384
  • 文/潘曉璐 我一進店門卧檐,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人焰宣,你說我怎么就攤上這事霉囚。” “怎么了匕积?”我有些...
    開封第一講書人閱讀 156,723評論 0 345
  • 文/不壞的土叔 我叫張陵盈罐,是天一觀的道長。 經(jīng)常有香客問我闪唆,道長盅粪,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 56,357評論 1 283
  • 正文 為了忘掉前任悄蕾,我火速辦了婚禮票顾,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己奠骄,他們只是感情好豆同,可當(dāng)我...
    茶點故事閱讀 65,412評論 5 384
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著含鳞,像睡著了一般影锈。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上民晒,一...
    開封第一講書人閱讀 49,760評論 1 289
  • 那天精居,我揣著相機與錄音,去河邊找鬼潜必。 笑死靴姿,一個胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 38,904評論 3 405
  • 文/蒼蘭香墨 我猛地睜開眼蛇更,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了维雇?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 37,672評論 0 266
  • 序言:老撾萬榮一對情侶失蹤晒他,失蹤者是張志新(化名)和其女友劉穎吱型,沒想到半個月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體陨仅,經(jīng)...
    沈念sama閱讀 44,118評論 1 303
  • 正文 獨居荒郊野嶺守林人離奇死亡津滞,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 36,456評論 2 325
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了灼伤。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片触徐。...
    茶點故事閱讀 38,599評論 1 340
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖狐赡,靈堂內(nèi)的尸體忽然破棺而出撞鹉,到底是詐尸還是另有隱情,我是刑警寧澤颖侄,帶...
    沈念sama閱讀 34,264評論 4 328
  • 正文 年R本政府宣布鸟雏,位于F島的核電站,受9級特大地震影響览祖,放射性物質(zhì)發(fā)生泄漏崔慧。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 39,857評論 3 312
  • 文/蒙蒙 一穴墅、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦玄货、人聲如沸皇钞。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,731評論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽夹界。三九已至,卻和暖如春隘世,著一層夾襖步出監(jiān)牢的瞬間可柿,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 31,956評論 1 264
  • 我被黑心中介騙來泰國打工丙者, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留复斥,地道東北人。 一個月前我還...
    沈念sama閱讀 46,286評論 2 360
  • 正文 我出身青樓械媒,卻偏偏與公主長得像目锭,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子纷捞,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 43,465評論 2 348

推薦閱讀更多精彩內(nèi)容

  • 單例模式 適用場景:可能會在場景中使用到對象痢虹,但只有一個實例,加載時并不主動創(chuàng)建主儡,需要時才創(chuàng)建 最常見的單例模式奖唯,...
    Obeing閱讀 2,056評論 1 10
  • 工廠模式類似于現(xiàn)實生活中的工廠可以產(chǎn)生大量相似的商品,去做同樣的事情糜值,實現(xiàn)同樣的效果;這時候需要使用工廠模式丰捷。簡單...
    舟漁行舟閱讀 7,724評論 2 17
  • 《ijs》速成開發(fā)手冊3.0 官方用戶交流:iApp開發(fā)交流(1) 239547050iApp開發(fā)交流(2) 10...
    葉染柒丶閱讀 5,089評論 0 7
  • 網(wǎng)站的劃分一般為二:前端和后臺。我們可以理解成后臺是用來實現(xiàn)網(wǎng)站的功能的臀玄,比如:實現(xiàn)用戶注冊瓢阴,用戶能夠為文章發(fā)表評...
    ConRon閱讀 755評論 0 0
  • 原文: https://github.com/ecomfe/spec/blob/master/javascript...
    zock閱讀 3,371評論 2 36