亚洲精品久久久中文字幕-亚洲精品久久片久久-亚洲精品久久青草-亚洲精品久久婷婷爱久久婷婷-亚洲精品久久午夜香蕉

您的位置:首頁技術(shù)文章
文章詳情頁

詳解JavaScript作用域 閉包

瀏覽:107日期:2023-10-19 15:57:58

JavaScript閉包,是JS開發(fā)工程師必須深入了解的知識(shí)。3月份自己曾撰寫博客《JavaScript閉包》,博客中只是簡單闡述了閉包的工作過程和列舉了幾個(gè)示例,并沒有去刨根問底,將其弄明白!

現(xiàn)在隨著對(duì)JavaScript更深入的了解,也剛讀完《你不知道的JavaScript(上卷)》這本書,所以乘機(jī)整理下,從底層和原理上去刨一下。

JavaScript并不具有動(dòng)態(tài)作用域,它只有詞法作用域。詞法作用域是在寫代碼或者說定義時(shí)確定的,而動(dòng)態(tài)作用域是在運(yùn)行時(shí)確定的。了解閉包前,首先我們得知道什么是詞法作用域(作用域是由書寫代碼時(shí)函數(shù)聲明的位置來決定的)。

一、何為閉包

示例1:

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

在foo()執(zhí)行后,通常認(rèn)為垃圾回收機(jī)制會(huì)將foo()的整個(gè)內(nèi)部作用域都被銷毀;而閉包可以阻止這樣事情發(fā)生,讓其內(nèi)部作用域依然存在。因?yàn)閎ar()處于foo()內(nèi)部,它擁有涵蓋foo()作用域的閉包,使得該作用域能夠一直存活,以供bar()在之后任何時(shí)間進(jìn)行引用。

bar()依然持有對(duì)該作用域的引用,而這個(gè)引用就叫作閉包。

簡言之:當(dāng)函數(shù)可以記住并訪問所在的詞法作用域,即使函數(shù)是在當(dāng)前詞法作用域之外執(zhí)行,這時(shí)就產(chǎn)生了閉包。

示例2:

無論使用何種方式對(duì)函數(shù)類型的值進(jìn)行傳遞,當(dāng)函數(shù)在別處被調(diào)用時(shí)都可以觀察到閉包。

function foo(){var a = 2;function baz(){console.log(a);}bar(baz);}function bar(fn){fn();// 這就是閉包}

示例3:

將一個(gè)內(nèi)部函數(shù)(timer)傳遞給setTimeout。timer具有涵蓋wait()作用域的閉包,保有對(duì)變量message的引用。

wait()執(zhí)行1000毫秒后,它的作用域并不會(huì)消失,timer依然保有wait()作用域的閉包。

function wait(message){setTimeout( function timer(){console.log(message);},1000);}wait('Hello,ligang');

示例4:

下述activator()具有涵蓋setupBot()作用域的閉包!

function setupBot(name, selector){$(selector).click(function activator(){console.log('Activating: '+ name);});}setupBot('Closure Bot 1', '#bot_1');setupBot('Closure Bot 2', '#bot_2');

二、循環(huán)和閉包

for(var i=1; i<=5; i++){setTimeout(function timer(){console.log(i);}, i*1000);}// 期望:每秒一次的頻率輸出1~5// 結(jié)果:每秒一次的頻率輸出五次6

先解釋一下:“i*1000”,5個(gè)定時(shí)分別在1s、2s、3s、4s、5s后執(zhí)行,并不是1s、3s、6s、10s、15s。也就是頻率為1s,不是每次間隔增加1s。如果去掉i寫成“1000”,會(huì)在for執(zhí)行完1s后直接輸出五次6。

回調(diào)函數(shù)在循環(huán)結(jié)束后才被執(zhí)行,因此輸出的是循環(huán)終止條件是i值。事實(shí)上,當(dāng)定時(shí)器運(yùn)行時(shí)即使每個(gè)迭代中執(zhí)行的是setTimeout(..., 0),所有的回調(diào)函數(shù)依然是在循環(huán)結(jié)束后才被執(zhí)行。

根據(jù)作用域的工作原理,盡管五個(gè)函數(shù)是在各個(gè)迭代中分別定義的,但是它們都被封閉在一個(gè)共享的全局作用域中,因此實(shí)際上只有一個(gè)i。

解決方案1:

for(var i=0; i<=5; i++){(function(j){setTimeout(function timer(){console.log(j);}, j*1000 );})(i);}// 結(jié)果:每秒一次的頻率輸出1~5

每個(gè)迭代都生成一個(gè)新的作用域,使得延遲函數(shù)的回調(diào)可以將新的作用封閉在每個(gè)迭代內(nèi)部,每個(gè)迭代中都會(huì)含有一個(gè)具有正確值的變量供我們?cè)L問。

解決方案2(ES6):

for(var i=0; i<=5; i++){let j = i;setTimeout(function timer(){console.log(j);}, j*1000 );}// 結(jié)果:每秒一次的頻率輸出1~5for(let i=0; i<=5; i++){setTimeout(function timer(){console.log(i);}, i*1000 );}// 結(jié)果:每秒一次的頻率輸出五次6

三、模塊

模塊需要具備兩個(gè)必要條件:

(1)必須有外部的封閉函數(shù),該函數(shù)必須至少被調(diào)用一次(每次調(diào)用都會(huì)創(chuàng)建一個(gè)新的模塊實(shí)例)。

(2)封閉函數(shù)必須返回至少一個(gè)內(nèi)部函數(shù),這樣內(nèi)部函數(shù)才能在私有作用域中形成閉包,并且可以訪問或者修改私有的狀態(tài)。

典型的模塊化:

function CoolMoudle(){var something = 'cool';var doSomething = function(){console.log(something);}return{doSomething: doSomething};}var foo = CoolMoudle();//如果不執(zhí)行外部函數(shù)CoolMoudle(),內(nèi)部作用域和閉包都無法創(chuàng)建foo.doSomething();//cool

單例模式:

var foo = (function CoolModule(id){function change(){// 修改公共APIpublicAPI.identify = identify2;}function identify1(){console.log(id);}function identify2(){console.log(id.toUpperCase());}var publicAPI = {change: change,identify: identify1};return publicAPI;})('foo module');foo.identify();//foo modulefoo.change();foo.identify(); //FOO MODULE

以上就是詳解JavaScript作用域 閉包的詳細(xì)內(nèi)容,更多關(guān)于JavaScript作用域 閉包的資料請(qǐng)關(guān)注好吧啦網(wǎng)其它相關(guān)文章!

標(biāo)簽: JavaScript
相關(guān)文章:
主站蜘蛛池模板: 在线播放精品 | 久久亚洲精品成人 | 狠狠狠色丁香婷婷综合久久五月 | 国产最爽的乱淫视频国语对 | 成年美女黄网站色视频大全免费 | 精品日韩欧美一区二区三区 | 欧美综合中文字幕久久 | 精品免费在线观看 | 亚洲精品永久一区 | 69国产成人综合久久精品91 | 免费黄视频在线观看 | 美女国产精品福利视频 | 台湾永久内衣秀130部vk | 一级做a爱片特黄在线观看 一级做a爱片特黄在线观看免费看 | 成人免费国产gav视频在线 | 99视频在线永久免费观看 | 中国美女一级毛片 | 1024你懂的国产欧美日韩在 | 国产精品亚欧美一区二区三区 | 91短视频版官网 | 视频在线观看黄 | 亚洲 国产 路线1路线2路线 | 日本欧美一区二区三区不卡视频 | 鲁大师7视频在线观看 | 黑人巨大解禁久久中文字幕 | 激情小视频在线播放免费 | 久爱青草视频在线观看 | 日韩欧美综合在线二区三区 | 未成人禁止视频高清在线观看 | 小明精品国产一区二区三区 | 国产精品你懂得 | 欧美成人精品福利在线视频 | 久久91久久91精品免费观看 | 在线观看日韩视频 | 久久亚洲国产成人影院 | 亚洲色图在线观看视频 | 青青青国产观看免费视频 | 伊人精品影院 | 91久久精一区二区三区大全 | 成人合成mv福利视频网站 | 美国一级大黄大色毛片 |