成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

JavaScript中的閉包

Donne / 426人閱讀

摘要:權(quán)威指南第版中閉包的定義函數(shù)對象可以通過作用域鏈相互關(guān)聯(lián)起來,函數(shù)體內(nèi)部的變量都可以保存在函數(shù)作用域內(nèi),這種特性在計算機(jī)科學(xué)文獻(xiàn)中成為閉包。循環(huán)中的閉包使用閉包時一種常見的錯誤情況是循環(huán)中的閉包,很多初學(xué)者都遇到了這個問題。

閉包簡介

閉包是JavaScript的重要特性,那么什么是閉包?

《JavaScript高級程序設(shè)計(第3版)》中閉包的定義:

閉包就是指有權(quán)訪問另一個函數(shù)中的變量的函數(shù)。

《JavaScript權(quán)威指南(第6版)》中閉包的定義:

函數(shù)對象可以通過作用域鏈相互關(guān)聯(lián)起來,函數(shù)體內(nèi)部的變量都可以保存在函數(shù)作用域內(nèi),這種特性在計算機(jī)科學(xué)文獻(xiàn)中成為“閉包”。

簡單來說,在JavaScript中,函數(shù)是對象,對象是屬性的集合,屬性的值也可以是對象,在函數(shù)內(nèi)定義函數(shù)就成為一種常見的情況,在函數(shù)內(nèi)部聲明函數(shù)innerFunction,在innerFunction內(nèi)部有權(quán)訪問外部函數(shù)的變量對象,這個函數(shù)就是我們所說的閉包。

我們來看一個簡單的例子:

function checkScope(){
    var scope = "local scope";
    function f() { return scope; }
    return f();
}
checkScope();//輸出為“l(fā)ocal scope”

當(dāng)函數(shù)第一次被調(diào)用時,會創(chuàng)建一個執(zhí)行環(huán)境以及相應(yīng)的作用域鏈,作用域鏈的前端,始終都是當(dāng)前執(zhí)行代碼所在環(huán)境的變量對象,作用域鏈中的下一個變量對象來自包含外部環(huán)境,下一個變量則來自下一個外部環(huán)境,這樣一直延續(xù)到全局執(zhí)行環(huán)境。

在上邊的例子中,訪問scope時,內(nèi)部的f()函數(shù)可以訪問f()外部的變量scope,因為它在作用域鏈中一級一級往上找的時候可以找到scope變量。

閉包的作用

一、 模擬私有變量。在函數(shù)內(nèi)創(chuàng)建一個閉包,閉包就可以通過自己的作用域鏈訪問函數(shù)內(nèi)部的變量,可以創(chuàng)建用于訪問私有變量的方法。訪問私有變量和私有函數(shù)的方法被稱為特權(quán)方法。

function MyObject(){
    var privateVariable = 10;
    function privateFunction() {
        return false;
    }
    //特權(quán)方法
    this.publicFunction = function() {
        privateVariable++;
        return privateFunction();
    };
}

二、 模仿塊級作用域。 JavaScript中沒有塊級作用域的概念,這意味著在塊語句中定義的變量,實際上是包含在函數(shù)中的。如果臨時需要一些變量,使用私有作用域。

function block() {
    var a = 1;
    var b = 2;
    (function () {
        var a = 3;//覆蓋了父作用域中的變量a
        var c = 4;
        //訪問到了當(dāng)前作用域中的變量
        console.log(a);//3
        //訪問了父作用域中的變量
        console.log(b);//2
        //訪問當(dāng)前作用域中的變量
        console.log(c);//4
    })()
    //訪問塊級作用域中的變量
    console.log(c);//c is not defined
}

這種技術(shù)經(jīng)常在全局作用域中被用在函數(shù)外部,從而限制向全局作用域中添加過多的變量和函數(shù)。

循環(huán)中的閉包

使用閉包時一種常見的錯誤情況是循環(huán)中的閉包,很多初學(xué)者都遇到了這個問題。很常見的一種情況就是給頁面中的多個按鈕綁定點擊事件,JavaScript代碼如下所示:

window.onload = function(){
    var inputs = document.getElementsByTagName("input");
    for(var i = 0; i < inputs.length; i++){
        inputs[i].onclick = function(){
            console.log(i);//希望輸出0,1,2,3,4...
        }
    }
}

頁面中有5個按鈕,根據(jù)上邊的代碼,我們需要的是依次點擊按鈕時,控制臺分別打印出0,1,2,3,4,而實際上,控制臺打印出來的,如下圖所示:

這是為什么呢,當(dāng)for循環(huán)執(zhí)行完之后,i已經(jīng)變成了按鈕的個數(shù)5了,而所有點擊函數(shù)綁定的都是同一個i,點擊按鈕時,打印出來的i也都變成了5了。

這一部分理解也可以參考http://www.cnblogs.com/qieguo...。

那么我們?yōu)榱说玫较胍慕Y(jié)果,需要在每次循環(huán)中創(chuàng)建變量i的拷貝,下面提供三種方法。

第一、使用匿名包裝器

window.onload = function () {
    var inputs = document.getElementsByTagName("input");
    for (var i = 0; i < inputs.length; i++) {
        (function (e) {
            inputs[i].onclick = function () {
                console.log(e);
            }
        })(i);

    }
}

依次點擊按鈕,控制臺輸出如下:

第二、從匿名包裝器中返回一個函數(shù):

window.onload = function () {
    var inputs = document.getElementsByTagName("input");
    for (var i = 0; i < inputs.length; i++) {
        inputs[i].onclick = function (num) {
            return function () {
                console.log(num);
            };
        } (i);
    }
}

首先,定義了匿名函數(shù),并將立即執(zhí)行該匿名函數(shù)的結(jié)果賦值給數(shù)組,匿名函數(shù)有一個參數(shù)num,在調(diào)用每個函數(shù)時,我們傳入了變量i,函數(shù)按值傳遞,就將變量i的當(dāng)前值復(fù)制給參數(shù)num。而在這個匿名函數(shù)內(nèi)部,又創(chuàng)建并返回了一個訪問num的閉包,這樣一來,每個按鈕點擊函數(shù)都有自己num變量的一個副本,因此可以輸出各自不同的數(shù)值了。

第三、在循環(huán)中使用let

window.onload = function () {
    var inputs = document.getElementsByTagName("input");
    for (let i = 0; i < inputs.length; i++) {
        inputs[i].onclick = function () {
            console.log(i);
        };
    }
}

let是ES6新增的命令,用法類似于var,但是所聲明的變量只能在let命令所在代碼塊內(nèi)有效。上述代碼中,變量i是let聲明的,當(dāng)前的i只在本輪循環(huán)有效。所以每一次循環(huán)的i其實都是一個新的變量。關(guān)于let的用法可參考《ECMAScript 6 入門》中第二章。

內(nèi)存泄漏

產(chǎn)生內(nèi)存泄漏的原因是IE9之前的版本對JScript對象和COM對象使用不同的垃圾收集例程,因此閉包在IE的這些版本中會導(dǎo)致一些問題。(JavaScript垃圾收集機(jī)制可參考《JavaScript高級程序設(shè)計(第3版)》4.3) 例如:

function assignHandle() {
    var element = document.getElementById("elementId");
    element.onclick = function () {
        //element的onclick引用了匿名函數(shù),
        //匿名函數(shù)又通過閉包引用了element,造成了循環(huán)引用
        console.log(element.id);
    };
}

這個例子中,循環(huán)引用導(dǎo)致了element引用數(shù)至少為1,element所占內(nèi)存就永遠(yuǎn)不會被回收,從而導(dǎo)致了內(nèi)存泄漏問題。要解決這個問題,就需要解除對DOM對象的引用,減少引用數(shù),確保正?;厥掌渌加玫膬?nèi)存。

引用計數(shù)的含義是跟蹤記錄每個值被引用的次數(shù)。當(dāng)聲明了一個變量并將一個引用類型值賦給該變量時,則這個值的引用次數(shù)就是1。如果同一個值又被賦給另一個變量,則該值的引用次數(shù)加1。相反,如果包含對這個值引用的變量又取得了另外一個值,則這個值的引用次數(shù)減1。當(dāng)這個值的引用次數(shù)變成0時,則說明沒有辦法再次訪問這個值了,因而就可以將其占用的內(nèi)存空間回收回來。

在采用引用計數(shù)策略的實現(xiàn)中,出現(xiàn)循環(huán)引用時,由于變量的引用次數(shù)永遠(yuǎn)不會是0,函數(shù)被多次調(diào)用時,就會導(dǎo)致大量內(nèi)存得不到回收。 這一部分理解可以參考MDN中JavaScript內(nèi)存管理。

結(jié)語

JavaScript閉包是極其有用的特性,但是由于閉包會攜帶包含它的函數(shù)的作用域,占用更多內(nèi)存,過多使用閉包可能會導(dǎo)致內(nèi)存占用過多。

文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/88242.html

相關(guān)文章

  • JavaScript中的閉包

    摘要:閉包引起的內(nèi)存泄漏總結(jié)從理論的角度將由于作用域鏈的特性中所有函數(shù)都是閉包但是從應(yīng)用的角度來說只有當(dāng)函數(shù)以返回值返回或者當(dāng)函數(shù)以參數(shù)形式使用或者當(dāng)函數(shù)中自由變量在函數(shù)外被引用時才能成為明確意義上的閉包。 文章同步到github js的閉包概念幾乎是任何面試官都會問的問題,最近把閉包這塊的概念梳理了一下,記錄成以下文章。 什么是閉包 我先列出一些官方及經(jīng)典書籍等書中給出的概念,這些概念雖然...

    HmyBmny 評論0 收藏0
  • JavaScript閉包,只學(xué)這篇就夠了

    摘要:當(dāng)在中調(diào)用匿名函數(shù)時,它們用的都是同一個閉包,而且在這個閉包中使用了和的當(dāng)前值的值為因為循環(huán)已經(jīng)結(jié)束,的值為。最好將閉包當(dāng)作是一個函數(shù)的入口創(chuàng)建的,而局部變量是被添加進(jìn)這個閉包的。 閉包不是魔法 這篇文章使用一些簡單的代碼例子來解釋JavaScript閉包的概念,即使新手也可以輕松參透閉包的含義。 其實只要理解了核心概念,閉包并不是那么的難于理解。但是,網(wǎng)上充斥了太多學(xué)術(shù)性的文章,對于...

    CoderBear 評論0 收藏0
  • 還擔(dān)心面試官問閉包?

    摘要:一言以蔽之,閉包,你就得掌握。當(dāng)函數(shù)記住并訪問所在的詞法作用域,閉包就產(chǎn)生了。所以閉包才會得以實現(xiàn)。從技術(shù)上講,這就是閉包。執(zhí)行后,他的內(nèi)部作用域并不會消失,函數(shù)依然保持有作用域的閉包。 網(wǎng)上總結(jié)閉包的文章已經(jīng)爛大街了,不敢說筆者這篇文章多么多么xxx,只是個人理解總結(jié)。各位看官瞅瞅就好,大神還希望多多指正。此篇文章總結(jié)與《JavaScript忍者秘籍》 《你不知道的JavaScri...

    tinyq 評論0 收藏0
  • 深入javascript——作用域和閉包

    摘要:注意由于閉包會額外的附帶函數(shù)的作用域內(nèi)部匿名函數(shù)攜帶外部函數(shù)的作用域,因此,閉包會比其它函數(shù)多占用些內(nèi)存空間,過度的使用可能會導(dǎo)致內(nèi)存占用的增加。 作用域和作用域鏈?zhǔn)莏avascript中非常重要的特性,對于他們的理解直接關(guān)系到對于整個javascript體系的理解,而閉包又是對作用域的延伸,也是在實際開發(fā)中經(jīng)常使用的一個特性,實際上,不僅僅是javascript,在很多語言中都...

    oogh 評論0 收藏0
  • Javascript閉包入門(譯文)

    摘要:也許最好的理解是閉包總是在進(jìn)入某個函數(shù)的時候被創(chuàng)建,而局部變量是被加入到這個閉包中。在函數(shù)內(nèi)部的函數(shù)的內(nèi)部聲明函數(shù)是可以的可以獲得不止一個層級的閉包。 前言 總括 :這篇文章使用有效的javascript代碼向程序員們解釋了閉包,大牛和功能型程序員請自行忽略。 譯者 :文章寫在2006年,可直到翻譯的21小時之前作者還在完善這篇文章,在Stackoverflow的How do Java...

    Fourierr 評論0 收藏0

發(fā)表評論

0條評論

最新活動
閱讀需要支付1元查看
<