欧美精品在线一区二区三区_亚洲女同精品视频_日韩一区免费_国产欧美久久久精品免费_国产这里只有精品_僵尸再翻生在线观看_久久99精品国产一区二区三区_亚洲免费一区二区_女教师淫辱の教室蜜臀av软件_中文字幕国产一区二区

JavaScript必須掌握的基礎 --- 閉包

2020-6-1    seo達人

閉包(Closure)的定義

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

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

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

ECMAScript中,閉包指的是:

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

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

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

閉包的優點:

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

閉包的缺點:

  1. 私有變量一直存在,占用內存。

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

自執行函數 ( IIFE )

自執行函數也叫立即調用函數(IIFE),是一個在定義時就執行的函數。

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

上述代碼是一個最簡單的自執行函數。

在ES6之前,是沒有塊級作用域的,只有全局作用域和函數作用域,因此自執行函數還能在ES6之前實現塊級作用域。

// 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沒有關系。如果我們把 let 換成 var ,就會污染全局變量 a 。

如果用自執行函數來實現:

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

為什么要在這里要引入自執行函數的概念呢?因為通常我們會用自執行函數來創建閉包,實現一定的效果。

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

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

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

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

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

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

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

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

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

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

那我們怎樣來修改它呢?

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

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

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

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

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

我們來優化一下:

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

*思路2:用 let 聲明變量,產生塊級作用域。

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

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

理論中的閉包

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

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

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

實踐中的閉包

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

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

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

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

我們先看個例子:

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

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

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

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

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

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

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

當checkscope執行碰到了 f 函數的創建,因此 f 函數也會創建對應的作用域鏈,默認以包裹它的函數執行時對應的作用域鏈為基礎。因此此時 f 函數創建時的作用域鏈如下:

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

當 f 函數執行時,此時的作用域鏈變化如下:

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

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

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

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

日歷

鏈接

個人資料

藍藍設計的小編 http://www.ocunn.cn

存檔

蜜桃tv一区二区三区| 国产成人美女视频| 一区二区三区免费在线观看视频| 国产精品免费精品一区| 三级理论午夜在线观看| 全球最大av网站久久| 韩国一区二区三区在线观看| 欧美激情在线一区二区| 日韩欧美一区二区三区在线| 青青精品视频播放| 三年中文高清在线观看第6集| 亚洲熟女一区二区三区| 中文字幕日本人妻久久久免费| 看黄网站在线观看| 神马久久一区二区三区| 国产区美女在线| 九九热爱视频精品视频| 国产一区二区三区四| 欧美性高潮床叫视频| 欧美猛交免费看| 亚洲欧美久久久久一区二区三区| 中文字幕无码人妻少妇免费| 国产视频第二页| 欧美大片免费高清观看| 亚洲精品美女91| 尤物视频一区二区| 久久夜色撩人精品| 永久免费精品视频网站| 亚洲欧美va天堂人熟伦| 成人c视频免费高清在线观看| 99er精品视频| 国产综合成人久久大片91| 91福利小视频| 国产精品美腿一区在线看| 欧美 激情 在线| 一本一道无码中文字幕精品热| 国产传媒在线播放| 伊人色**天天综合婷婷| 亚洲欧美aⅴ...| 欧美精品在线免费播放| 欧美 国产 精品| 青青草激情视频| 免费网站黄在线观看| 欧美hentaied在线观看| 亚洲欧美影音先锋| 美女av一区二区三区| 9色视频在线观看| 国产午夜小视频| av在线播放观看| 欧美福利电影在线观看| 亚洲www啪成人一区二区麻豆| 欧美激情亚洲自拍| 欧美性久久久久| 中文字幕乱伦视频| 91成人抖音| 国产麻豆视频精品| 亚洲黄色免费三级| 日韩中文字幕一区| 欧美日韩偷拍视频| 成人三级小说| 免费成人小视频| 日韩一级黄色片| 久久手机视频| 开心激情五月网| 国产素人视频在线观看| 亚洲精品乱码| 欧美精品 国产精品| 岛国一区二区三区高清视频| 黄色正能量网站| 国产视频精品久久| 欧美日韩国产精品一区二区亚洲| 婷婷六月综合亚洲| 成人黄色免费在线观看| 91视频啊啊啊| 三区四区在线视频| 亚洲精品免费观看| 日韩一区二区电影在线| 麻豆av一区二区| 九九热精品免费视频| a级大胆欧美人体大胆666| 久操视频在线观看免费 | 蜜臀久久99精品久久久久宅男| 91精品免费观看| 欧美日韩国产不卡在线看| 免费无码毛片一区二区app| 日本黄色免费在线| 国产成人在线观看| 日韩视频在线免费观看| 国产二区视频在线播放| 精品久久久久成人码免费动漫| 福利片一区二区| 自拍偷拍亚洲欧美日韩| 日韩免费av片在线观看| 久久久国产精品无码| 日韩免费网站| 免费精品视频在线| 亚洲香蕉成视频在线观看| 欧美精品一区二区三区三州| www日本高清| 色婷婷亚洲mv天堂mv在影片| 一本久道久久综合中文字幕| 国产日韩一区二区| 亚洲激情视频一区| 成人国产精品久久| 亚洲欧洲精品一区二区三区不卡 | 永久免费精品影视网站| 激情网站五月天| 黄动漫在线看| 亚洲黄色免费| 日韩成人中文电影| 女性隐私黄www网站视频| 91网在线观看| 亚洲免费黄色| 亚洲欧美综合v| 别急慢慢来1978如如2| a4yy在线播放免费观看视频| 黄色一区二区三区四区| 精品人伦一区二区色婷婷| 免费日韩在线观看| 狠狠人妻久久久久久综合麻豆| 亚洲电影影音先锋| 精品国产百合女同互慰| xxxx18hd亚洲hd捆绑| 最近最好的中文字幕2019免费 | 精品国产乱码久久久久久1区2匹| 欧美综合在线视频观看| 国产视频一区欧美| 日韩不卡中文字幕| 毛片毛片毛片毛片毛片毛片毛片毛片毛片 | 55夜色66夜色国产精品视频| www国产91| 国产婷婷一区二区三区| 高清免费观看在线| 老司机亚洲精品| 综合激情国产一区| 欧美一区二区三区影院| 国产在线xxx| 国产人成亚洲第一网站在线播放| 国产欧美亚洲精品| 国产一级片免费观看| 国产成人一二片| 欧美视频一区二区三区| 99久热在线精品视频| 在线资源av| 久久精品久久精品| 2020久久国产精品| 午夜爽爽爽男女免费观看| 久久wwww| 欧美无乱码久久久免费午夜一区| 高清无码一区二区在线观看吞精| 一本到av在线| 激情综合亚洲精品| 国产91久久婷婷一区二区| 精品一区在线观看视频| 久久99国产精品久久99大师| 欧美揉bbbbb揉bbbbb| 亚洲中文字幕无码av永久| 精品三级久久久久久久电影聊斋| 国产精品66部| 国产精品香蕉av| 亚洲 欧美 另类人妖| 亚洲永久无码7777kkk| 欧美大片黄色| 国产精品高潮久久久久无| 激情视频一区二区| 成人av手机在线| 影音先锋男人资源在线| 久久精品国产久精国产| 国产97在线|日韩| 好吊妞视频一区二区三区| 色偷偷综合网| 中文欧美日本在线资源| 成人午夜剧场视频网站| 91精品国产自产在线丝袜啪 | 欧美三级午夜理伦| 亚洲欧美综合久久久| 国产一区二区欧美日韩| 国产特黄级aaaaa片免| 日韩欧美中文字幕在线视频| 91精品国产入口| 日本中文字幕观看| 在线观看v片| 色婷婷综合五月| 国产1区2区在线| 免费毛片b在线观看| 性做久久久久久| 无码人妻精品一区二区三区在线| av色综合久久天堂av色综合在| 一色桃子久久精品亚洲| 宅男在线精品国产免费观看| 国产小视频免费在线观看| 国产网站一区二区| 一区二区三区不卡在线| 成人资源www网在线最新版| 中文字幕欧美激情一区| 中文字幕99| 香蕉视频在线看| 亚洲男人的天堂网| 国产美女主播在线| 欧美wwww| 欧美性猛交xxxx偷拍洗澡| 欧美aⅴ在线观看| 原纱央莉成人av片| 欧美亚洲国产一区在线观看网站| 五月婷婷六月合| 成人在线视频观看| 日韩一区二区精品| 国产福利在线观看视频| 私拍精品福利视频在线一区| 国产亚洲人成网站在线观看| 国产三级精品三级观看| 久久久久蜜桃| 欧美一级大片在线观看| 一区二区日韩在线观看| 国产一区在线视频| 久久久久久久久久久一区| 日韩有码电影| 亚洲免费av高清| 黄色片久久久久| 久久99国产精品二区高清软件| 日韩女优视频免费观看| 在线国产视频一区| 青青草91久久久久久久久| 欧美激情一二三| 亚洲熟妇无码久久精品| 国产一区啦啦啦在线观看| 久久久久se| 免费av不卡| 欧洲视频一区二区| 精品国产av色一区二区深夜久久| 一区三区在线欧| 欧美精品九九久久| 91欧美日韩麻豆精品| www.在线成人| 天天爱天天做天天操| 国产桃色电影在线播放| 欧美精品亚洲二区| 精品成人无码一区二区三区| 在线免费观看日本欧美爱情大片| 日韩av色综合| 宅男午夜电影| 亚洲免费资源在线播放| 九九热精品在线播放| 久久精品色综合| 欧美极品欧美精品欧美视频| 国产精品视频第一页| 久久综合色一综合色88| 黄色一级片在线看| 一区二区三区| www.久久色.com| 中文字幕精品在线观看| 99久久久久免费精品国产| 欧美中文字幕在线观看视频| jizz久久久久久| 中文字幕亚洲精品| 国产精品高清无码| 97精品久久久久中文字幕| av在线播放天堂| 成人豆花视频| 欧美成人精品激情在线观看| 91影院在线播放| 国产日韩精品一区二区三区 | 欧美 日韩 激情| 久久丁香四色| 久热精品视频在线| 涩涩视频在线观看| wwwwww.欧美系列| 一女被多男玩喷潮视频| 中文字幕一区二区三区中文字幕| 欧美久久精品午夜青青大伊人| 99久久久久成人国产免费| 国产三级精品三级在线专区| 亚洲中文字幕久久精品无码喷水| 久久99偷拍| 人人爽久久涩噜噜噜网站| 国产69精品久久久久毛片| 亚洲精品在线91| 日韩精品国产一区二区| 久久99精品国产.久久久久久 | 成人在线视频亚洲| 欧美一区2区视频在线观看| 91麻豆精品成人一区二区| 日韩高清电影一区| 亚洲图片都市激情| 日本成人在线网站| 久久91亚洲人成电影网站| 草草视频在线播放| 亚洲一区二区av电影| 亚洲av人人澡人人爽人人夜夜| 亚洲精品99| 久久99精品久久久久久青青日本 | 中文字幕久精品免费视频| www.97av.com| av在线免费观看网站| 一区二区成人在线| 色婷婷激情一区二区三区| 99热国产免费| 亚洲国产精品精华素| 亚洲国产福利在线| 中文字幕永久在线视频| 国产精品成人免费精品自在线观看| 亚洲男人天堂av在线| 综合天堂久久久久久久| 久久综合中文色婷婷| 日韩高清中文字幕一区二区| 久久亚洲国产精品成人av秋霞| 三级视频在线看| 在线观看一区二区视频| 国产大学生自拍| 99麻豆久久久国产精品免费 | 在线免费看91| 一区二区三区91| a级黄色免费视频| 国产成人一级电影| 在线观看免费黄网站| 中文在线播放一区二区| 色一情一乱一伦一区二区三区| 国产精品视频在线观看免费| 久久青草久久| 四虎精品欧美一区二区免费| 国产劲爆久久| 91精品视频播放| 国产在线88av| 久久97久久97精品免视看| 四虎精品在线| 亚洲午夜在线| 蜜桃麻豆91| 成人av在线播放| 国产精品日韩欧美大师| av资源一区| 欧美激情中文字幕在线| 国产视频福利在线| 亚洲免费av电影| 乱精品一区字幕二区| 欧美日韩夫妻久久| 亚洲中文无码av在线| 亚洲午夜免费电影| 免费在线观看黄视频| 国产精品乱码久久久久久| 波多野吉衣中文字幕| 成人免费观看av| 折磨小男生性器羞耻的故事| 美女在线视频一区| 一道本视频在线观看| 久久国产成人| 十八禁视频网站在线观看| 亚洲国产高清一区| 69堂免费视频| 亚洲经典在线看| 日本a级片免费观看| 亚洲美女少妇无套啪啪呻吟| 国产毛片视频网站| 尤物在线精品| 日日摸日日碰夜夜爽av| 99精品国产一区二区青青牛奶 | 成人交换视频| 国产精品久久久| 久久青草伊人| 日本欧美黄网站| 深夜成人影院| 国产精品亚发布| 黄色成人在线视频| 亚洲一区二区三区成人在线视频精品| 国产精品亚洲一区二区三区在线观看 | 精品亚洲精品福利线在观看| 在线成人性视频| 精品国产一区二区三区不卡蜜臂 | 国产精品444| 国产一区二区三区朝在线观看| 国产欧美日韩91| 电影中文字幕一区二区| 国产视频一区二区三区四区| 久久a爱视频| 一级做a爰片久久| 国产免费无码一区二区视频| 中文乱码免费一区二区| 欧美爱爱免费视频| 亚洲黄色录像片| 国产女主播喷水视频在线观看| 色婷婷av一区二区三区之一色屋| 中文在线观看av| 91精品国产综合久久国产大片| 少妇无码一区二区三区| 亚洲男人天堂视频| av在线日韩国产精品| 97免费中文视频在线观看| 成人欧美一区二区三区的电影| 国产日韩精品在线播放| 天堂精品久久久久| 相泽南亚洲一区二区在线播放| 中文字幕亚洲精品乱码| 免费观看成人在线视频| 久久91精品国产91久久小草| 免费a在线观看播放| 中文av一区特黄| 最近免费中文字幕大全免费版视频| 欧美日韩美女一区二区| 宅男深夜视频| 欧美日韩福利电影| 日韩一区二区三区在线免费观看|