99久久精品国产片-99久久精品国产免费-99久久精品国产麻豆-99久久精品国产国产毛片-99久久精品国产高清一区二区-99久久精品费精品国产一区二区

JavaScript必須掌握的基礎(chǔ) --- 閉包

2020-6-1    seo達人

閉包(Closure)的定義

閉包是一個讓初級JavaScript使用者既熟悉又陌生的一個概念。因為閉包在我們書寫JavaScript代碼時,隨處可見,但是我們又不知道哪里用了閉包。

關(guān)于閉包的定義,網(wǎng)上(書上)的解釋總是千奇百怪,我們也只能“取其精華去其糟粕”去總結(jié)一下。

  1. 即使函數(shù)在當前作用域外調(diào)用,但是還能訪問當前作用域中的變量和函數(shù)
  2. 有權(quán)訪問另一個函數(shù)作用域中的變量(函數(shù))的函數(shù)。
  3. 閉包是指那些能夠訪問自由變量的函數(shù)

ECMAScript中,閉包指的是:

  1. 從理論角度:所有的函數(shù)都是閉包。因為它們都在創(chuàng)建的時候就將上層上下文的數(shù)據(jù)保存起來了。哪怕是簡單的全局變量也是如此,因為函數(shù)中訪問全局變量也就相當于是在訪問自由變量,這個時候使用最外層的作用域。
  2. 從實踐角度:一下才算是閉包:

    • 即使創(chuàng)建它的上下文已經(jīng)銷毀,它仍然存在。
    • 在代碼中引用了自由變量。

閉包跟詞法作用域,作用域鏈,執(zhí)行上下文這幾個JavaScript中重要的概念都有關(guān)系,因此要想真的理解閉包,至少要對那幾個概念不陌生。

閉包的優(yōu)點:

  1. 可以是用函數(shù)內(nèi)部的變量(函數(shù)),也可以說是可以訪問函數(shù)作用域。
  2. 擁有私有變量,避免污染全局變量

閉包的缺點:

  1. 私有變量一直存在,占用內(nèi)存。

我們來一步一步引出閉包。

自執(zhí)行函數(shù) ( IIFE )

自執(zhí)行函數(shù)也叫立即調(diào)用函數(shù)(IIFE),是一個在定義時就執(zhí)行的函數(shù)。

var a=1;
(function() { console.log(a)
})()

上述代碼是一個最簡單的自執(zhí)行函數(shù)。

在ES6之前,是沒有塊級作用域的,只有全局作用域和函數(shù)作用域,因此自執(zhí)行函數(shù)還能在ES6之前實現(xiàn)塊級作用域。

// ES6 塊級作用域 var a = 1; if(true) { let a=111; console.log(a); // 111 } console.log(a); // 1 

這里 if{} 中用let聲明了一個 a。這個 a 就具有塊級作用域,在這個 {} 中訪問 a ,永遠訪問的都是 let 聲明的a,跟全局作用域中的a沒有關(guān)系。如果我們把 let 換成 var ,就會污染全局變量 a 。

如果用自執(zhí)行函數(shù)來實現(xiàn):

var a = 1;
(function() { if(true) { var a=111; console.log(a); // 111 }
})() console.log(a); // 1

為什么要在這里要引入自執(zhí)行函數(shù)的概念呢?因為通常我們會用自執(zhí)行函數(shù)來創(chuàng)建閉包,實現(xiàn)一定的效果。

來看一個基本上面試提問題:

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

在理想狀態(tài)下我們期望輸出的是 0 ,1 ,2 ,3 ,4。但是實際上輸出的是5 ,5 ,5 ,5 ,5。為什么是這樣呢?其實這里不僅僅涉及到作用域,作用域鏈還涉及到Event Loop、微任務、宏任務。但是在這里不講這些。

下面我們先解釋它為什么會輸出 5個5,然后再用自執(zhí)行函數(shù)來修改它,以達到我們預期的結(jié)果。

提示:for 循環(huán)中,每一次的都聲明一個同名變量,下一個變量的值為上一次循環(huán)執(zhí)行完同名變量的值。

首先用var聲明變量 for 是不會產(chǎn)生塊級作用域的,所以在 () 中聲明的 i 為全局變量。相當于:

// 偽代碼 var i; for(i=0;i<5;i++) {
    setTimeout(function() { console.log(i);
    },1000)
}

setTimeout中的第一個參數(shù)為一個全局的匿名函數(shù)。相當于:

// 偽代碼 var i; var f = function() { console.log(i);
} for(i=0;i<5;i++) {
    setTimeout(f,1000)
}

由于setTimeout是在1秒之后執(zhí)行的,這個時候for循環(huán)已經(jīng)執(zhí)行完畢,此時的全局變量 i 已經(jīng)變成了 5 。1秒后5個setTimeout中的匿名函數(shù)會同時執(zhí)行,也就是5個 f 函數(shù)執(zhí)行。這個時候 f 函數(shù)使用的變量 i 根據(jù)作用域鏈的查找規(guī)則找到了全局作用域中的 i 。因此會輸出 5 個5。

那我們怎樣來修改它呢?

  • 思路1:讓setTimeout匿名函數(shù)中訪問的變量 i 不再訪問全局作用域中的 i 。因此把它包裹在一個函數(shù)作用域中。這時 匿名函數(shù)訪問變量 i 時,會先去包裹它的函數(shù)作用域中查找。
for(var i=0;i<5;i++) {
    (function (){ setTimeout(function() { console.log(i);
        },1000)
    })();
}

上述例子會輸出我們期望的值嗎?答案是否。為什么呢?我們雖然把 setTimeout 包裹在一個匿名函數(shù)中了,但是當setTimeout中匿名函數(shù)執(zhí)行時,首先去匿名函數(shù)中查找 i 的值,找不到還是會找到全局作用域中,最終 i 的值仍然是全局變量中的 i ,仍然為 5個5.

那我們把外層的匿名函數(shù)中聲明一個變量 j 讓setTimeout中的匿名函數(shù)訪問這個 j 不就找不到全局變量中的變量了嗎。

for(var i=0;i<5;i++) {
    (function (){ var j = i;
        setTimeout(function() { console.log(j);
        },1000)
    })();
}

這個時候才達到了我們預期的結(jié)果:0 1 2 3 4。

我們來優(yōu)化一下:

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

*思路2:用 let 聲明變量,產(chǎn)生塊級作用域。

for(let i=0;i<5;i++) {
    setTimeout(function() { console.log(i);
    },1000)
}

這時for循環(huán)5次,產(chǎn)生 5 個塊級作用域,也會聲明 5 個具有塊級作用域的變量 i ,因此setTimeout中的匿名函數(shù)每次執(zhí)行時,訪問的 i 都是當前塊級作用域中的變量 i 。

理論中的閉包

什么是理論中的閉包?就是看似像閉包,其實并不是閉包。它只是類似于閉包。

 function foo() { var a=2; function bar() { console.log(a); // 2 }
    bar();
}
foo();

上述代碼根據(jù)最上面我們對閉包的定義,它并不完全是閉包,雖然是一個函數(shù)可以訪問另一個函數(shù)中的變量,但是被嵌套的函數(shù)是在當前詞法作用域中被調(diào)用的。

實踐中的閉包

我們怎樣把上述代碼foo 函數(shù)中的bar函數(shù),在它所在的詞法作用域外執(zhí)行呢?

下面的代碼就清晰的展示了閉包:

function foo() { var a=2; function bar() { console.log(a);
    } return bar;
} var baz=foo();
baz(); // 2 —— 朋友,這就是閉包的效果。

上述代碼中 bar 被當做 foo函數(shù)返回值。foo函數(shù)執(zhí)行后把返回值也就是 bar函數(shù) 賦值給了全局變量 baz。當 baz 執(zhí)行時,實際上也就是 bar 函數(shù)的執(zhí)行。我們知道 foo 函數(shù)在執(zhí)行后,foo 的內(nèi)部作用域會被銷毀,因為引擎有垃圾回收期來釋放不再使用的內(nèi)存空間。所以在bar函數(shù)執(zhí)行時,實際上foo函數(shù)內(nèi)部的作用域已經(jīng)不存在了,理應來說 bar函數(shù) 內(nèi)部再訪問 a 變量時是找不到的。但是閉包的神奇之處就在這里。由于 bar 是在 foo 作用域中被聲明的,所以 bar函數(shù) 會一直保存著對 foo 作用域的引用。這時就形成了閉包。

我們先看個例子:

var scope = "global scope"; function checkscope(){ var scope = "local scope"; function f(){ return scope;
    } return f;
} var foo = checkscope();
foo();

我們用偽代碼來解釋JavaScript引擎在執(zhí)行上述代碼時的步驟:

  1. JavaScript引擎遇到可執(zhí)行代碼時,就會進入一個執(zhí)行上下文(環(huán)境)
  2. 首先遇到的是全局代碼,因此進入全局執(zhí)行上下文,把全局執(zhí)行上下文壓入執(zhí)行上下文棧。
  3. 全局上下文創(chuàng)建時會先在內(nèi)部創(chuàng)建VO/AO,作用域鏈,this。然后執(zhí)行代碼。
  4. 當遇到 checkscope 函數(shù)執(zhí)行時,進入checkscope的執(zhí)行上下文,然后壓入執(zhí)行上下文棧。
  5. checkscope 執(zhí)行上下文創(chuàng)建時會先在內(nèi)部創(chuàng)建VO/AO,作用域鏈,this。然后執(zhí)行代碼。
  6. 當checkscope 函數(shù)執(zhí)行完畢時,會從執(zhí)行上下文棧中彈出,此時它的AO也會被瀏覽器回收。(這是理想狀態(tài)下)
  7. 執(zhí)行foo函數(shù),向上查找foo的值,發(fā)現(xiàn)foo的值為checkscope函數(shù)內(nèi)部函數(shù)f。因此這一步為執(zhí)行 checkscope 內(nèi)部函數(shù)f。
  8. 執(zhí)行f函數(shù)同執(zhí)行 checkscope 的步驟一致。
  9. f 函數(shù)執(zhí)行完畢,從執(zhí)行上下文棧中彈出。

但是我們想一個問題,checkscope函數(shù)執(zhí)行完畢,它的執(zhí)行上下文從棧中彈出,也就是銷毀了不存在了,f 函數(shù)還能訪問包裹函數(shù)的作用域中的變量(scope)嗎?答案是可以。

理由是在第6步,我們說過當checkscope 執(zhí)行函數(shù)執(zhí)行完畢時,它的執(zhí)行上下文會從棧中彈出,此時活動對象也會被回收,按理說當 f 在訪問checkscope的活動對象時是訪問不到的。

其實這里還有個概念,叫做作用域鏈:當 checkscope 函數(shù)被創(chuàng)建時,會創(chuàng)建對應的作用域鏈,里面值存放著包裹它的作用域?qū)獔?zhí)行上下文的變量對象,在這里只是全局執(zhí)行上下文的變量對象,當checkscope執(zhí)行時,此時的作用域鏈變化了 ,里面存放的是變量對象(活動對象)的集合,最頂端是當前函數(shù)的執(zhí)行上下文的活動對象。端是全局執(zhí)行上下文的變量對象。類似于:

checkscope.scopeChain = [
    checkscope.AO
    global.VO
] 

當checkscope執(zhí)行碰到了 f 函數(shù)的創(chuàng)建,因此 f 函數(shù)也會創(chuàng)建對應的作用域鏈,默認以包裹它的函數(shù)執(zhí)行時對應的作用域鏈為基礎(chǔ)。因此此時 f 函數(shù)創(chuàng)建時的作用域鏈如下:

checkscope.scopeChain = [
    checkscope.AO
    global.VO
]

當 f 函數(shù)執(zhí)行時,此時的作用域鏈變化如下:

checkscope.scopeChain = [
    f.AO
    checkscope.AO
    global.VO
]

當checkscope函數(shù)執(zhí)行完畢,內(nèi)部作用域會被回收,但是 f函數(shù) 的作用域鏈還是存在的,里面存放著 checkscope函數(shù)的活動對象,因此在f函數(shù)執(zhí)行時會從作用域鏈中查找內(nèi)部使用的 scope 標識符,從而在作用域鏈的第二位找到了,也就是在 checkscope.AO 找到了變量scope的值。

正是因為JavaScript做到了這一點,因此才會有閉包的概念。還有人說閉包并不是為了擁有它采取設(shè)計它的,而是設(shè)計作用域鏈時的副作用產(chǎn)物。

閉包是JavaScript中最難的點,也是平常面試中常問的問題,我們必須要真正的去理解它,如果只靠死記硬背是經(jīng)不起考驗的。

日歷

鏈接

個人資料

藍藍設(shè)計的小編 http://www.lapeinture.cn

存檔

日韩中文字幕一区| 国产亚洲免费观看| 国产一级生活片| 99热精品在线| 国产激情视频在线观看| 99久久视频| 一级女性全黄生活片免费| 成人免费福利片在线观看| 日本伦理黄色大片在线观看网站| 国产一区免费观看| 亚洲精品中文字幕久久久久久| 四虎影视久久久| 国产麻豆精品hdvideoss| 黄视频网站在线看| 久草免费在线观看| 九九久久国产精品大片| 日韩综合| 亚洲精品久久久中文字| 精品国产一区二区三区久久久蜜臀| 麻豆网站在线看| 夜夜操网| 久久99爰这里有精品国产| 国产国语对白一级毛片| 沈樵在线观看福利| 国产原创中文字幕| 韩国毛片 免费| 国产视频一区二区在线播放| 成人a大片在线观看| 欧美夜夜骑 青草视频在线观看完整版 久久精品99无色码中文字幕 欧美日韩一区二区在线观看视频 欧美中文字幕在线视频 www.99精品 香蕉视频久久 | 国产不卡福利| 一级毛片看真人在线视频| 日韩在线观看免费| a级毛片免费观看网站| 亚洲天堂免费| 成人高清视频免费观看| 麻豆午夜视频| 欧美α片无限看在线观看免费| 日本在线www| 欧美a级v片不卡在线观看| 国产一区免费观看| 99久久视频| 99久久精品国产片| 99色视频在线| 午夜在线亚洲| 精品国产香蕉在线播出| 国产网站免费观看| 日韩在线观看免费完整版视频| 日本特黄一级| 国产成人精品综合在线| 精品视频免费观看| a级黄色毛片免费播放视频| 天天做人人爱夜夜爽2020毛片| 日本在线www| 久久久久久久免费视频| 精品毛片视频| 夜夜操网| 成人免费观看男女羞羞视频| 国产高清在线精品一区二区| 99久久精品国产麻豆| 国产精品自拍一区| 精品久久久久久综合网| 国产伦精品一区三区视频| 精品久久久久久中文字幕一区| 欧美激情在线精品video| 久久精品大片| 精品国产亚洲人成在线| 成人免费高清视频| 99热精品在线| 欧美国产日韩一区二区三区| 日本免费乱人伦在线观看| 91麻豆爱豆果冻天美星空| 国产91精品一区| 欧美夜夜骑 青草视频在线观看完整版 久久精品99无色码中文字幕 欧美日韩一区二区在线观看视频 欧美中文字幕在线视频 www.99精品 香蕉视频久久 | 黄视频网站免费看| 精品国产一区二区三区久| 二级片在线观看| 精品国产一区二区三区久久久狼| 黄视频网站免费观看| 欧美a级片视频| 美女免费毛片| 久久精品店| 国产麻豆精品高清在线播放| 日韩免费片| 亚洲www美色| 国产不卡在线看| 午夜在线影院| 国产网站免费| a级毛片免费观看网站| 天天做人人爱夜夜爽2020| 日韩中文字幕在线播放| 亚洲第一视频在线播放| 九九干| 日日日夜夜操| 久久精品免视看国产明星| 美国一区二区三区| 深夜做爰性大片中文| 高清一级做a爱过程不卡视频| 中文字幕一区二区三区 精品| 午夜激情视频在线观看| 精品国产一区二区三区免费| 国产视频久久久| 精品国产亚一区二区三区| 精品在线视频播放| 91麻豆国产福利精品| 国产伦久视频免费观看 视频| 青草国产在线观看| 成人免费福利片在线观看| 青青青草视频在线观看| 欧美夜夜骑 青草视频在线观看完整版 久久精品99无色码中文字幕 欧美日韩一区二区在线观看视频 欧美中文字幕在线视频 www.99精品 香蕉视频久久 | 精品久久久久久综合网| 成人免费观看网欧美片| 91麻豆精品国产片在线观看| 精品在线免费播放| 久久成人综合网| 日韩中文字幕一区二区不卡| 九九久久国产精品大片| 久久福利影视| 美国一区二区三区| 免费国产在线观看不卡| 韩国三级视频网站| 沈樵在线观看福利| 精品国产亚一区二区三区| 天天色色网| 欧美国产日韩在线| 国产网站免费| 日日夜夜婷婷| 97视频免费在线观看| 免费毛片播放| 精品视频在线看 | 九九热精品免费观看| 国产成人女人在线视频观看| 亚洲www美色| 精品久久久久久中文字幕2017| 国产不卡在线看| 欧美大片一区| 999精品视频在线| 好男人天堂网 久久精品国产这里是免费 国产精品成人一区二区 男人天堂网2021 男人的天堂在线观看 丁香六月综合激情 | 台湾美女古装一级毛片| 一级毛片看真人在线视频| 午夜家庭影院| 欧美18性精品| 日本免费乱人伦在线观看| 日日日夜夜操| 久久国产影院| 99久久网站| a级精品九九九大片免费看| 国产美女在线一区二区三区| 久久国产精品只做精品| 二级特黄绝大片免费视频大片| 久草免费资源| 日本免费乱理伦片在线观看2018| 成人影院一区二区三区| 精品久久久久久综合网| 国产一区二区精品久久91| 国产视频一区在线| 亚洲 欧美 成人日韩| 国产成人精品影视| 九九热国产视频| 国产一区二区精品久久| 可以免费看毛片的网站| 高清一级毛片一本到免费观看| 国产成人啪精品视频免费软件| 国产一区二区精品尤物| 一本高清在线| 欧美另类videosbestsex高清| 午夜久久网| 国产一区二区精品尤物| 麻豆网站在线看| 国产成人女人在线视频观看| 国产a毛片| 久久国产影院| 午夜精品国产自在现线拍| 日本伦理网站| 精品国产亚一区二区三区| 亚洲精品中文一区不卡| 国产a免费观看| 精品毛片视频| 国产伦精品一区二区三区无广告| 国产一区二区精品| 亚洲精品影院| 青草国产在线观看| 欧美激情一区二区三区在线| 国产国语在线播放视频| 91麻豆精品国产自产在线 | 精品久久久久久免费影院| 一级毛片视频播放| 久久国产精品永久免费网站| 成人免费观看网欧美片| 久久国产影院| 日韩在线观看免费| 成人高清视频免费观看| 精品久久久久久综合网| 91麻豆爱豆果冻天美星空| 国产韩国精品一区二区三区| 国产伦精品一区二区三区无广告 | 青草国产在线观看| 99色吧| 久久福利影视| 国产网站在线| 国产不卡福利|