国产xxxx99真实实拍_久久不雅视频_高清韩国a级特黄毛片_嗯老师别我我受不了了小说

資訊專欄INFORMATION COLUMN

【重溫基礎(chǔ)】19.閉包

nanfeiyan / 2481人閱讀

摘要:系列目錄復(fù)習(xí)資料資料整理個(gè)人整理重溫基礎(chǔ)篇重溫基礎(chǔ)對(duì)象介紹重溫基礎(chǔ)對(duì)象介紹重溫基礎(chǔ)介紹重溫基礎(chǔ)相等性判斷本章節(jié)復(fù)習(xí)的是中的關(guān)于閉包,這個(gè)小哥哥呀,看看。這里隨著閉包函數(shù)的結(jié)束,執(zhí)行環(huán)境銷毀,變量回收。

本文是 重溫基礎(chǔ) 系列文章的第十九篇。
今日感受:將混亂的事情找出之間的聯(lián)系,也是種能力。

系列目錄:

【復(fù)習(xí)資料】ES6/ES7/ES8/ES9資料整理(個(gè)人整理)

【重溫基礎(chǔ)】1-14篇

【重溫基礎(chǔ)】15.JS對(duì)象介紹

【重溫基礎(chǔ)】16.JSON對(duì)象介紹

【重溫基礎(chǔ)】17.WebAPI介紹

【重溫基礎(chǔ)】18.相等性判斷

本章節(jié)復(fù)習(xí)的是JS中的關(guān)于閉包,這個(gè)小哥哥呀,看看。

前置知識(shí):
聲明函數(shù)兩種方法:

函數(shù)聲明,存在函數(shù)聲明提升,因此可以在函數(shù)聲明之前調(diào)用(不會(huì)報(bào)錯(cuò))。

fun();  // ok
function fun(){};

函數(shù)表達(dá)式,不存在函數(shù)聲明提升,若定義前調(diào)用,會(huì)報(bào)錯(cuò)(函數(shù)還不存在)。

fun();  // error
var fun = function (){};
1.概念 2.1 詞法作用域

這里先要了解一個(gè)概念,詞法作用域:它是靜態(tài)的作用域,是書寫變量和塊作用域的作用域**。

function f (){
    var a = "leo";
    function g(){console.log(a)};
    g();
}
f(); // "leo"

由于函數(shù)g的作用域中沒有a這個(gè)變量,但是它可以訪問(wèn)父作用域,并使用父作用域下的變量a,最后輸出"leo"。

詞法作用域中使用的域,是變量在代碼中聲明的位置所決定的。嵌套的函數(shù)可以訪問(wèn)在其外部聲明的變量。

2.2 閉包

接下來(lái)介紹下閉包概念,閉包是指有權(quán)訪問(wèn)另一個(gè)函數(shù)作用域中的變量的函數(shù)

閉包是由函數(shù)以及創(chuàng)建該函數(shù)的詞法環(huán)境組合而成。這個(gè)環(huán)境包含了這個(gè)閉包創(chuàng)建時(shí)所能訪問(wèn)的所有局部變量。

創(chuàng)建閉包的常見方式:在一個(gè)函數(shù)內(nèi)創(chuàng)建另一個(gè)函數(shù)。如:

function f (){
    var a = "leo";
    var g = function (){
        console.log(a);
    };
    return g;// 這里g就是一個(gè)閉包函數(shù),可以訪問(wèn)到g作用域的變量a
}
var fun = f();
fun(); // "leo"

通過(guò)概念可以看出,閉包有以下三個(gè)特征:

函數(shù)嵌套函數(shù)

函數(shù)內(nèi)部可以引用函數(shù)外部的參數(shù)和變量

參數(shù)和變量不會(huì)被垃圾回收機(jī)制回收

注:關(guān)于內(nèi)存回收機(jī)制,可以查看阮一峰老師的《JavaScript 內(nèi)存泄漏教程》。

另外,使用閉包有以下好處:

將一個(gè)變量長(zhǎng)期保存在內(nèi)存中

避免全局變量的污染

function f (){
    var a = 1; 
    return function(){
        a++;
        console.log(a);
    }
}
var fun = f();
fun(); // 2
fun(); // 3

因?yàn)槔厥諜C(jī)制沒有回收,所以每次調(diào)用fun()都會(huì)返回新的值。

私有化成員,使得外部不能訪問(wèn)

function f (){
    var a = 1;
    function f1 (){
        a++;
        console.log(a);
    };
    function f2 (){
        a++;
        console.log(a);
    };
    return {g1:f1, g2:f2};
};
var fun = f();
fun.g1(); // 2
fun.g2(); // 3
2.易錯(cuò)點(diǎn) 2.1 引用的變量發(fā)生變化
function f (){
    var a = [];
    for(var i = 0; i<10; i++){
        a[i] = function(){
            console.log(i);
        }
    }
    return a;
}
var fun = f();
fun[0]();  // 10
fun[1]();  // 10
// ...
fun[10]();  // 10

原本照我們的想法,fun方法中每個(gè)元素上的方法執(zhí)行的結(jié)果應(yīng)該是1,2,3,...,10,而實(shí)際上,每個(gè)返回都是10,因?yàn)槊總€(gè)閉包函數(shù)引用的變量if執(zhí)行環(huán)境下的變量i,循環(huán)結(jié)束后,i已經(jīng)變成10,所以都會(huì)返回10
解決辦法可以這樣:

function f (){
    var a = [];
    for(var i = 0; i<10; i++){
        a[i] = function(index){
            return function(){
                console.log(index);
                // 此時(shí)的index,是父函數(shù)作用域的index,
                // 數(shù)組的10個(gè)函數(shù)對(duì)象,每個(gè)對(duì)象的執(zhí)行環(huán)境下的index都不同
            }
        }(i);
    };
    return a;
};
var fun = f();
fun[0]();  // 0
fun[1]();  // 1
// ...
fun[10]();  // 10
2.2 this指向問(wèn)題
var obj = {
    name : "leo", 
    f : function(){
        return function(){
            console.log(this.name);
        }
    }
}
obj.f()();  // undefined

由于里面的閉包函數(shù)是在window作用域下執(zhí)行,因此this指向window

2.3 內(nèi)存泄漏

當(dāng)我們?cè)陂]包內(nèi)引用父作用域的變量,會(huì)使得變量無(wú)法被回收。

function f (){
    var a = document.getElementById("leo");
    a.onclick = function(){console.log(a.id)};
}

這樣做的話,變量a會(huì)一直存在無(wú)法釋放,類似的變量越來(lái)越多的話,很容易引起內(nèi)存泄漏。我們可以這么解決:

function f (){
    var a = document.getElementById("leo");
    var id = a.id;
    a.onclick = function(){};
    a = null;  //主動(dòng)釋放變量a
}

通過(guò)把變量賦值成null來(lái)主動(dòng)釋放掉。

3.案例 3.1 經(jīng)典案例——定時(shí)器和閉包

代碼如下:

for(var i = 0 ; i<10; i++){
    setTimeout(function(){
        console.log(i);
    },100);
}

不出所料,返回的不是我們想要的0,1,2,3,...,9,而是10個(gè)10。
這是因?yàn)閖s是單進(jìn)程,所以在執(zhí)行for循環(huán)的時(shí)候定時(shí)器setTimeout被安排到任務(wù)隊(duì)列中排隊(duì)等候執(zhí)行,而在等待過(guò)程中,for循環(huán)已經(jīng)在執(zhí)行,等到setTimeout要執(zhí)行的時(shí)候,for循環(huán)已經(jīng)執(zhí)行完成,i的值就是10,所以就打印了10個(gè)10。
解決方法 :

1.使用ES6新增的let

for循環(huán)中的var替換成let。

2.使用閉包

for(var i = 0; i<10 ; i++){
    (function(i){
        setTimeout(function(){
            console.log(i);
        }, i*100);
    })(i);
}
3.2 使用閉包解決遞歸調(diào)用問(wèn)題
function f(num){
    return num >1 ? num*f(num-1) : 1;
}

var fun = f;
f = null;
fun(4)   // 報(bào)錯(cuò) ,因?yàn)樽詈檬莚eturn num* arguments.callee(num-1),arguments.callee指向當(dāng)前執(zhí)行函數(shù),但是在嚴(yán)格模式下不能使用該屬性也會(huì)報(bào)錯(cuò),所以借助閉包來(lái)實(shí)現(xiàn)

這里可以使用return num >1 ? num* arguments.callee(num-1) : 1;,因?yàn)?b>arguments.callee指向當(dāng)前執(zhí)行函數(shù),但是在嚴(yán)格模式下不能使用,也會(huì)報(bào)錯(cuò),所以這里需要使用閉包來(lái)實(shí)現(xiàn)。

function fun = (function f(num){
    return num >1 ? num*f(num-1) : 1;
})

這樣做,實(shí)際上起作用的是閉包函數(shù)f,而不是外面的fun。

3.3 使用閉包模仿塊級(jí)作用域

ES6之前,使用var聲明變量會(huì)有變量提升問(wèn)題:

for(var i = 0 ; i<10; i++){console.log(i)};
console.log(i);  // 變量提升 返回10

為了避免這個(gè)問(wèn)題,我們這樣使用閉包(匿名自執(zhí)行函數(shù)):

(function(){
    for(var i = 0 ; i<10; i++){console.log(i)};
})()
console.log(i);  // undefined

我們創(chuàng)建了一個(gè)匿名的函數(shù),并立即執(zhí)行它,由于外部無(wú)法引用它內(nèi)部的變量,因此在函數(shù)執(zhí)行完后會(huì)立刻釋放資源,關(guān)鍵是不污染全局對(duì)象。這里i隨著閉包函數(shù)的結(jié)束,執(zhí)行環(huán)境銷毀,變量回收。
但是現(xiàn)在,我們用的更多的是ES6規(guī)范的letconst來(lái)聲明。

參考文章

MDN 閉包

《JavaScript高級(jí)程序設(shè)計(jì)》

本部分內(nèi)容到這結(jié)束

Author 王平安
E-mail pingan8787@qq.com
博 客 www.pingan8787.com
微 信 pingan8787
每日文章推薦 https://github.com/pingan8787...
JS小冊(cè) js.pingan8787.com

歡迎關(guān)注微信公眾號(hào)【前端自習(xí)課】每天早晨,與您一起學(xué)習(xí)一篇優(yōu)秀的前端技術(shù)博文 .

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

轉(zhuǎn)載請(qǐng)注明本文地址:http://m.specialneedsforspecialkids.com/yun/101370.html

相關(guān)文章

  • 重溫基礎(chǔ)】22.內(nèi)存管理

    摘要:內(nèi)存泄露內(nèi)存泄露概念在計(jì)算機(jī)科學(xué)中,內(nèi)存泄漏指由于疏忽或錯(cuò)誤造成程序未能釋放已經(jīng)不再使用的內(nèi)存。判斷內(nèi)存泄漏,以字段為準(zhǔn)。 本文是 重溫基礎(chǔ) 系列文章的第二十二篇。 今日感受:優(yōu)化學(xué)習(xí)方法。 系列目錄: 【復(fù)習(xí)資料】ES6/ES7/ES8/ES9資料整理(個(gè)人整理) 【重溫基礎(chǔ)】1-14篇 【重溫基礎(chǔ)】15.JS對(duì)象介紹 【重溫基礎(chǔ)】16.JSON對(duì)象介紹 【重溫基礎(chǔ)】1...

    Pandaaa 評(píng)論0 收藏0
  • 重溫基礎(chǔ)】4.函數(shù)

    摘要:本文是重溫基礎(chǔ)系列文章的第四篇。系列目錄復(fù)習(xí)資料資料整理個(gè)人整理重溫基礎(chǔ)語(yǔ)法和數(shù)據(jù)類型重溫基礎(chǔ)流程控制和錯(cuò)誤處理重溫基礎(chǔ)循環(huán)和迭代本章節(jié)復(fù)習(xí)的是中的基礎(chǔ)組件之一,函數(shù),用來(lái)復(fù)用特定執(zhí)行邏輯。箭頭函數(shù)不能使用命令,即不能用作函數(shù)。 本文是 重溫基礎(chǔ) 系列文章的第四篇。今日感受:常懷感恩之心,對(duì)人對(duì)己。 系列目錄: 【復(fù)習(xí)資料】ES6/ES7/ES8/ES9資料整理(個(gè)人整理) 【重溫基...

    maxmin 評(píng)論0 收藏0
  • 重溫基礎(chǔ)】21.高階函數(shù)

    摘要:歡迎您的支持系列目錄復(fù)習(xí)資料資料整理個(gè)人整理重溫基礎(chǔ)篇重溫基礎(chǔ)對(duì)象介紹重溫基礎(chǔ)對(duì)象介紹重溫基礎(chǔ)介紹重溫基礎(chǔ)相等性判斷重溫基礎(chǔ)閉包重溫基礎(chǔ)事件本章節(jié)復(fù)習(xí)的是中的高階函數(shù),可以提高我們的開發(fā)效率。 本文是 重溫基礎(chǔ) 系列文章的第二十一篇。 今日感受:想家。 本人自己整理的【Cute-JavaScript】資料,包含:【ES6/ES7/ES8/ES9】,【JavaScript基礎(chǔ)...

    wua_wua2012 評(píng)論0 收藏0
  • 重溫基礎(chǔ)】20.事件

    摘要:本文是重溫基礎(chǔ)系列文章的第二十篇。事件捕獲為截獲事件提供機(jī)會(huì),然后實(shí)際的目標(biāo)接收到事件,最后事件冒泡,對(duì)事件作出響應(yīng)。事件處理事件處理,即響應(yīng)某個(gè)事件。包括導(dǎo)致事件的元素事件類型等其他信息。 本文是 重溫基礎(chǔ) 系列文章的第二十篇。 這是第三個(gè)基礎(chǔ)系列的第一篇,歡迎持續(xù)關(guān)注呀! 重溫基礎(chǔ) 系列的【初級(jí)】和【中級(jí)】的文章,已經(jīng)統(tǒng)一整理到我的【Cute-JavaScript】的Java...

    Blackjun 評(píng)論0 收藏0
  • Javascript重溫OOP之作用域與閉包

    摘要:的變量作用域是基于其特有的作用域鏈的。需要注意的是,用創(chuàng)建的函數(shù),其作用域指向全局作用域。所以,有另一種說(shuō)法認(rèn)為閉包是由函數(shù)和與其相關(guān)的引用環(huán)境組合而成的實(shí)體。 作用域 定義 在編程語(yǔ)言中,作用域控制著變量與參數(shù)的可見性及生命周期,它能減少名稱沖突,而且提供了自動(dòng)內(nèi)存管理 --javascript 語(yǔ)言精粹 我理解的是,一個(gè)變量、函數(shù)或者成員可以在代碼中訪問(wèn)到的范圍。 js的變量作...

    JessYanCoding 評(píng)論0 收藏0

發(fā)表評(píng)論

0條評(píng)論

最新活動(dòng)
閱讀需要支付1元查看
<