最近朋友面試被問到了 JS 閉包的問題,本人一時(shí)語塞,想起了袁華的一句話:“這道題太難了,我不會(huì)做,不會(huì)做啊!”。
成都創(chuàng)新互聯(lián)公司專注于企業(yè)營(yíng)銷型網(wǎng)站、網(wǎng)站重做改版、東興網(wǎng)站定制設(shè)計(jì)、自適應(yīng)品牌網(wǎng)站建設(shè)、HTML5、商城網(wǎng)站制作、集團(tuán)公司官網(wǎng)建設(shè)、成都外貿(mào)網(wǎng)站建設(shè)公司、高端網(wǎng)站制作、響應(yīng)式網(wǎng)頁設(shè)計(jì)等建站業(yè)務(wù),價(jià)格優(yōu)惠性價(jià)比高,為東興等各大城市提供網(wǎng)站開發(fā)制作服務(wù)。
JS 閉包屬于面向?qū)ο蟮囊粋€(gè)重要知識(shí)點(diǎn),特此本人又開始了一段說走就走的旅程。
閉包就是外層函數(shù)的作用域(AO)對(duì)象被內(nèi)層函數(shù)所引用,無法被釋放。
上面那句話聽起來可能不是很理解,本人在之前寫過一篇Python 閉包小記》的關(guān)于 Python 閉包的一些知識(shí)的文章,里面寫了百度百科對(duì)于閉包的理解,雖然由于才疏學(xué)淺大部分都是引用的他人的知識(shí)架構(gòu),但語言這種東西都是相通的,我們不需要去記那些晦澀的名詞,對(duì)于閉包,作為初學(xué)者我們只需知道:
函數(shù)作為返回值,函數(shù)作為參數(shù)傳遞。就可以將其理解為閉包。
話不多說,先上個(gè)代碼緩和一下尷尬的氣氛:
function outer() { var max = 10; function inner(num) { if (num > max) { console.log(num) } } return inner; } var foo = outer(); foo(20); // 20
上面代碼滿足函數(shù)作為返回值的條件,所以是一個(gè)閉包函數(shù)。
根據(jù) JS 函數(shù)的執(zhí)行機(jī)制,先執(zhí)行第 10 行的 foo 代碼,在函數(shù)執(zhí)行完之后會(huì)被 JS 的垃圾回收機(jī)制將 outer 函數(shù)回收,但是在執(zhí)行到第 3 行的時(shí)候我們發(fā)現(xiàn) outer 函數(shù)內(nèi)部又出現(xiàn)了一個(gè) inner 函數(shù),且 inner 函數(shù)里引用著 outer 函數(shù)的 max = 10; 的變量,這就無法被回收并且留在了內(nèi)存里,當(dāng)執(zhí)行到第 11 行時(shí)由于 outer 函數(shù)內(nèi)的 max = 10; 被留在內(nèi)存中,所以會(huì)被 inner 函數(shù)調(diào)用,并滿足 if 條件判斷,所以輸出 20;
以上我們實(shí)現(xiàn)了一個(gè)簡(jiǎn)單的閉包函數(shù),但是卻產(chǎn)生了一個(gè)問題,那就是無法被釋放的對(duì)象留在了內(nèi)存當(dāng)中,造成了不必要的內(nèi)存開銷。
再看如下代碼:
var max = 10, foo = function (num) { if (num > max) { console.log(num); } }; (function (bar) { var max = 100; bar(20) })(foo); // 20
上面代碼滿足函數(shù)作為參數(shù)傳遞的條件,所以是一個(gè)閉包函數(shù)。
函數(shù) foo 作為一個(gè)參數(shù)被傳入函數(shù)中,賦值個(gè) bar 參數(shù),當(dāng)執(zhí)行到 bar() 函數(shù)時(shí),函數(shù)內(nèi)部的 max 并不是 100,而是 10,這似乎匪夷所思。我們暫且將 7 — 10 行的函數(shù)叫 “父作用域”,其余叫“全局作用域”,當(dāng)執(zhí)行到 bar(20) 時(shí),函數(shù)去執(zhí)行第 2 行的代碼,此時(shí) foo 函數(shù)內(nèi)部的 max 要去取值,而 max = 10; 正好在他所在的 “全局作用域” 內(nèi),所以會(huì)取 max = 10; 的值而不是 max = 100; 的值。由此可見,取值時(shí)要去創(chuàng)造這個(gè)函數(shù)的作用域內(nèi)取值,而不是所謂的 “父作用域” 或者離函數(shù)近的地方取值。
我們?cè)賮砜匆欢未a:
var num = 20; function outer() { var max = 10; function inner() { if (num > max) { console.log(num); } } return inner; } var foo = outer(), num = 30; foo(); //30
上面的代碼在看完上面的解釋后可以得知它是一個(gè)閉包函數(shù),且定義了一個(gè)全局變量 num,最初定義為 num = 20,當(dāng)代碼執(zhí)行到第 11 行時(shí)去調(diào)用執(zhí)行第 2 行,待第 11 行執(zhí)行完畢后執(zhí)行第 12 行,此時(shí)將全局的 num = 20; 變?yōu)榱?num = 30; 再執(zhí)行第 13 行,此時(shí)執(zhí)行時(shí)調(diào)用 inner 函數(shù)時(shí),從輸出結(jié)果我們可以看出調(diào)用的 num 為之后賦值的 30,
由此可見全局的 num 變量被污染了。
我們?cè)賮砜聪乱欢未a:
function outer() { var max = 10; function inner(num) { if (num > max) { console.log(num); } } return inner; } var foo = outer(), max = 100; foo(20); //20
上面的代碼中當(dāng)函數(shù)執(zhí)行時(shí),先執(zhí)行第 10 行,然后調(diào)用執(zhí)行第 1 行的函數(shù),此時(shí)將 max 賦值為 10,但需要注意的是此時(shí)的 max = 10;并不是在全局作用域內(nèi),而是在 outer() 函數(shù)的作用域內(nèi),執(zhí)行完第 10 行再執(zhí)行第 11 行,此時(shí)將 max 賦值為 100,但需要注意的是此時(shí)的 max = 100;是在全局作用域內(nèi)。所以在執(zhí)行到第 12 行代碼的時(shí)候調(diào)用執(zhí)行 inner() 函數(shù)并將參數(shù) 20 傳入,輸出結(jié)果為 20,由此可見outer() 函數(shù)作用域內(nèi)的對(duì)象 num 并沒有被全局的對(duì)象 num 所污染。
由以上四段代碼我們初步了解了一些閉包的基本特征,但是由于才疏學(xué)淺,怕總結(jié)的不夠全面,這時(shí)突然想到了東東大神的筆記,于是上網(wǎng)搜到了一些,下面就將其再歸納總結(jié)一下。
閉包:既重用一個(gè)變量,又保護(hù)變量不被污染的一種機(jī)制。
為什么使用閉包:
全局變量和局部變量都具有不可兼得的優(yōu)缺點(diǎn)。
全局變量:
局部變量:
何時(shí)使用:
只要既重用一個(gè)變量,又保護(hù)變量不被污染時(shí)。
如何使用:
閉包形成的原因:
外層函數(shù)調(diào)用后,外層函數(shù)的函數(shù)作用域(AO)對(duì)象無法釋放,被內(nèi)層函數(shù)引用著。
閉包的缺點(diǎn):
結(jié)合上面舉的四段代碼栗子和東東的筆記,我們已經(jīng)對(duì)閉包有了一個(gè)形象的認(rèn)識(shí),但是要到達(dá)全面理解的程度,只能說革命尚未成功,同志仍需努力。
令人可喜的是在網(wǎng)上又查到了東東對(duì)于閉包更形象的圖形講解,看完之后相信大家對(duì)閉包會(huì)有更加深刻的理解。
先來一段代碼緩和一下字多的尷尬:
//1. 用外層函數(shù)包裹要保護(hù)的變量和內(nèi)層函數(shù) function outer() { var i = 1; //2. 外層函數(shù)返回內(nèi)層函數(shù)對(duì)象到外部 return function () { console.log(i++); } } //3. 調(diào)用外層函數(shù)獲得內(nèi)層函數(shù)對(duì)象 var getNum = outer(); //getNum:function(){ console.log(i++); } getNum();//1 getNum();//2 i = 1; getNum();//3 getNum();//4
上面的代碼是定義了一個(gè) outer() 外層函數(shù),外層函數(shù)的作用域內(nèi)定義了 i = 1;的變量,內(nèi)部返回了一個(gè)函數(shù),這就形成了閉包。當(dāng)代碼執(zhí)行到第 10 行,其實(shí)就返回了一個(gè) outer() 函數(shù)的內(nèi)部函數(shù),執(zhí)行一次 getNum(),由于打印的是 i++ ,所以輸出結(jié)果為 1,(注:如果打印的是 ++i,輸出結(jié)果為 2 )。再執(zhí)行一次 getNum(),由于之前 i 已經(jīng)執(zhí)行過一次 i++,所以此次執(zhí)行結(jié)果為 2,再在全局設(shè)置 i = 1,再次執(zhí)行 getNum() 兩次,執(zhí)行結(jié)果分別為 3 和 4,說明全局設(shè)置的 i = 1,并沒有覆蓋 outer() 函數(shù)作用域內(nèi)的 i 值,outer() 函數(shù)內(nèi)的 i 值被很好的保護(hù)起來并得到了重用。
我們來看看東東對(duì)上面代碼的圖形化分析:
如上圖:在 JavaScript 中有一個(gè)執(zhí)行環(huán)境棧(ECS)概念,注:ECS = 局部EC + 全局EC,所有的函數(shù)都要通過進(jìn)棧、出棧來執(zhí)行,執(zhí)行環(huán)境棧中有一個(gè)自帶的 main() 函數(shù)的全局EC 指向全局的 window 作用域,它會(huì)指向全局的 window 對(duì)象,代碼運(yùn)行到紅線部分的時(shí)候,執(zhí)行環(huán)境棧中僅有一個(gè)全局執(zhí)行環(huán)境 window,此時(shí) window 中有兩個(gè)全局變量(標(biāo)識(shí)符):outer 、getNum,其中 outer() 函數(shù)開辟了一塊內(nèi)存用于存儲(chǔ)所執(zhí)行的方法,并且通過 scope 記住它的父級(jí)。
如上圖:當(dāng)執(zhí)行 outer() 函數(shù)時(shí),outer() 相當(dāng)于局部EC 進(jìn)入執(zhí)行環(huán)境棧,此時(shí) outer() 會(huì)開辟一塊屬于自己的作用域(AO),里面定義了 i = 1,的環(huán)境變量。 由于 window 中引用著 i 對(duì)象,所以 outer 的 AO 會(huì)指向 window,同時(shí) getNum 會(huì)調(diào)用 outer() 函數(shù)并返回一個(gè)方法,所以會(huì)開辟一塊內(nèi)存用于存儲(chǔ)所執(zhí)行的方法,該方法中又有 i 變量指向 outer 的 AO,綠色線三方互相牽連。
如上圖:當(dāng)執(zhí)行環(huán)境棧中的 outer() 函數(shù)執(zhí)行完出棧時(shí),理論上 outer 的 AO,即藍(lán)色框應(yīng)該被垃圾回收機(jī)制所回收,但是由于閉包作用,這塊就被留了下來,閉包至此形成。
如上圖:當(dāng) outer() 函數(shù)出棧,getNum() 函數(shù)進(jìn)棧,getNum 開辟屬于自己的作用域(AO),且執(zhí)行了一次 i++ 。此時(shí)輸出結(jié)果為 1。
如上圖:當(dāng) getNum() 函數(shù)出棧時(shí),自己多開辟的作用域被回收,但是 outer 的作用域由于閉包作用依然留在內(nèi)存中,且變?yōu)榱?i = 2。
如上圖:再次執(zhí)行 getNum() 函數(shù),相當(dāng)于 getNum() 函數(shù)再次入棧出棧,原來由于閉包作用保留的 i = 2 再次做 ++ 運(yùn)算。
如上圖:再往下執(zhí)行 i = 1,即在全局 window 當(dāng)中添加了 i 對(duì)象。此時(shí) outer 作用域內(nèi)的 i 由于上一次的 ++ 變?yōu)榱?3。
如上圖:第三次執(zhí)行 getNum() 函數(shù),此時(shí)大家應(yīng)該懂得該怎么執(zhí)行了吧,getNum() 并不會(huì)去全局的 window 中去取 i = 1 使用,而是去所創(chuàng)造它的作用域去值,即 i = 3 做 ++ 運(yùn)算。
至此閉包的運(yùn)行流程就全部介紹完了,大家是不是對(duì)于閉包有了一個(gè)比較清晰的了解了。
別急,還差那么一點(diǎn)點(diǎn),那就是主動(dòng)釋放閉包所產(chǎn)生的內(nèi)存。如下
//1. 用外層函數(shù)包裹要保護(hù)的變量和內(nèi)層函數(shù) function outer() { var i = 1; //2. 外層函數(shù)返回內(nèi)層函數(shù)對(duì)象到外部 return function () { console.log(i++); i = null; } } //3. 調(diào)用外層函數(shù)獲得內(nèi)層函數(shù)對(duì)象 var getNum = outer(); //getNum:function(){ console.log(i++); } getNum(); //1 getNum(); //0 i = 1; getNum(); //0 getNum(); //0
在執(zhí)行完第一次 getNum() 函數(shù)時(shí)我們就將 i 變量設(shè)為 null,再次執(zhí)行 getNum() 函數(shù)時(shí)發(fā)現(xiàn)所得結(jié)果已經(jīng)變?yōu)?0 了,說明 outer() 函數(shù)內(nèi)的 i 變量?jī)?nèi)存已經(jīng)被釋放了?。?!
至此 JavaScript 閉包的全部?jī)?nèi)容就講解完畢了,以上內(nèi)容如有紕漏請(qǐng)各位大神批評(píng)指正。
好記性不如爛筆頭,特此記錄,與君共勉!
以上所述是小編給大家介紹的JavaScript閉包詳解整合,希望對(duì)大家有所幫助,如果大家有任何疑問請(qǐng)給我留言,小編會(huì)及時(shí)回復(fù)大家的。在此也非常感謝大家對(duì)創(chuàng)新互聯(lián)網(wǎng)站的支持!