閉包:從字面意思來(lái)理解就是封閉和包裹挺峡,換句話說(shuō)担钮,就是在函數(shù)內(nèi)部定義的變量箫津,在函數(shù)的外部無(wú)法訪問(wèn)到狭姨,因此就說(shuō)函數(shù)構(gòu)成了一個(gè)閉包。
說(shuō)白了閉包是作用域的應(yīng)用苏遥。
計(jì)算機(jī)科學(xué)中對(duì)閉包的定義為:一個(gè)函數(shù)的函數(shù)體以及函數(shù)所處的環(huán)境饼拍,構(gòu)成的一個(gè)綜合體叫做:閉包
函數(shù)體:就是函數(shù)內(nèi)部的代碼。
函數(shù)所處的環(huán)境:指的就是作用域田炭。
《Javascript高級(jí)程序設(shè)計(jì)》上對(duì)閉包的定義是:有權(quán)限訪問(wèn)另一個(gè)函數(shù)作用域中的變量的函數(shù)师抄。也就是說(shuō),閉包是一個(gè)函數(shù)教硫,那什么樣的函數(shù)才能是閉包呢叨吮?他能訪問(wèn)另一個(gè)函數(shù)作用域中的變量。這樣的解釋讓我們直接想起了一個(gè)函數(shù)的內(nèi)部函數(shù)栋豫,因?yàn)楦鶕?jù)作用域鏈的規(guī)則挤安,只有嵌套的函數(shù)才能達(dá)到這個(gè)效果丧鸯。并且這個(gè)閉包函數(shù)是作為父函數(shù)的返回值返回蛤铜,而且這個(gè)閉包函數(shù)通常是個(gè)匿名函數(shù)。
使用閉包來(lái)解決什么樣的問(wèn)題丛肢?
我使用閉包來(lái)解決如何訪問(wèn)函數(shù)內(nèi)部變量的問(wèn)題围肥。
function foo() {
var num = 123;
function fn() {
return num;//
}
return fn;
}
var m=foo();
var f=m();
console.log(f);//輸出的值為123,這樣就可以訪問(wèn)到函數(shù)內(nèi)部變量num蜂怎。
原理就是利用閉包:在函數(shù)foo中嵌套了一個(gè)函數(shù)fn,利用函數(shù)fn可以訪問(wèn)到其外部的變量穆刻,而獲取函數(shù)foo中的變量num,當(dāng)外部調(diào)用函數(shù)foo()時(shí)候, 函數(shù)foo中 return返回fn函數(shù)的實(shí)體杠步,然后我再對(duì)函數(shù)fn進(jìn)行調(diào)用就在外部訪問(wèn)到函數(shù)內(nèi)部變量num了氢伟。就是利用內(nèi)部函數(shù)的閉包特性把函數(shù)foo中的變量取到外邊, 說(shuō)白了內(nèi)部函數(shù)就相當(dāng)于一個(gè)媒介幽歼,就是一個(gè)橋梁連接函數(shù)內(nèi)部和外部朵锣。這就是閉包作用。
《高級(jí)程序設(shè)計(jì)》上甸私,這樣說(shuō):當(dāng)在函數(shù)內(nèi)部定義了其他函數(shù)時(shí)候诚些,就創(chuàng)建了閉包。閉包有權(quán)訪問(wèn)包含函數(shù)內(nèi)部的所有變量皇型。
(這句話怎么理解呢诬烹?照這句話理解的話砸烦,閉包就是一個(gè)嵌套函數(shù)嘛!嵌套函數(shù)對(duì)包含它的函數(shù)的變量當(dāng)然可以訪問(wèn)绞吁,這是沒(méi)有問(wèn)題的幢痘。)
一般來(lái)說(shuō),內(nèi)部函數(shù)是能夠訪問(wèn)到上一級(jí)乃至全局的的變量的掀泳,那么就有人這樣理解:通過(guò)閉包雪隧,可以實(shí)現(xiàn)外部訪問(wèn)函數(shù)局部?jī)?nèi)的變量。
(如果我們把作用域簡(jiǎn)單的分個(gè)級(jí)的話员舵,假設(shè)全局作用域作為第一級(jí)脑沿,其中定義的函數(shù)體內(nèi)部作用域作為第二級(jí),在第二級(jí)作用域內(nèi)嵌套定義的函數(shù)體內(nèi)部作用域作為第三級(jí)马僻,....等等庄拇,傳統(tǒng)意義上,第一級(jí)不能訪問(wèn)第二級(jí)的變量(這種變量叫做局部變量)韭邓,第二級(jí)不能訪問(wèn)第三級(jí)措近,...,而反過(guò)來(lái)是可以的女淑,這就是作用域鏈瞭郑。本級(jí)作用域內(nèi)找不到再到上一級(jí)找,直至第一級(jí)全局鸭你。而閉包這種機(jī)制可以在第一級(jí)作用域中通過(guò)第三級(jí)作用域引用到第二級(jí)作用域中的變量屈张,而方法就是在第二級(jí)作用域向第一級(jí)作用域返回?fù)碛械谌?jí)作用域的函數(shù)引用。 這個(gè)引用才是關(guān)鍵袱巨,因?yàn)檫@個(gè)引用的存在阁谆,相關(guān)的第三作用域與第二作用域都成了這個(gè)引用運(yùn)行的上下文,迫使垃圾回收機(jī)制GC不能回收這條鏈上所占用的資源愉老。而如果沒(méi)有這個(gè)引用场绿,則跟一般函數(shù)一樣,函數(shù)運(yùn)行完資源就會(huì)被回收嫉入。而我的疑惑也在于此焰盗,閉包單指函數(shù)中的嵌套函數(shù)還是指被第一級(jí)引用了的嵌套函數(shù)?還是都是咒林?還是說(shuō)閉包并不是嵌套函數(shù)而是嵌套函數(shù)被第一級(jí)作用域引用時(shí)所形成的這種機(jī)制姨谷?)
實(shí)際上是就是閉包延長(zhǎng)變量的生命周期。通常函數(shù)的作用域即變量會(huì)在函數(shù)執(zhí)行結(jié)束后被銷(xiāo)毀映九,但當(dāng)函數(shù)返回一個(gè)閉包,只要閉包不被釋放瞎颗,整條作用域鏈都會(huì)占用內(nèi)存件甥。(閉包延長(zhǎng)變量的生命周期捌议,這是指被第一級(jí)引用的情況。但如果沒(méi)有這個(gè)引用引有,閉包還能稱(chēng)其為閉包嗎瓣颅?)
說(shuō)道作用域鏈:即 函數(shù)自己的作用域、上一層的函數(shù)的作用域....和全局作用域譬正。訪問(wèn)一個(gè)變量時(shí)宫补,自己的沒(méi)有,就一層層往上找曾我,直至全局粉怕,若還沒(méi)有,就報(bào)錯(cuò)抒巢。
閉包是一個(gè)概念贫贝,它描述了函數(shù)執(zhí)行完畢內(nèi)存釋放后,依然內(nèi)存駐留的一個(gè)現(xiàn)象蛉谜,只要把握這個(gè)核心概念稚晚,閉包就不難理解了。