什么是閉包?
函數(shù)嵌套函數(shù)棋恼,內(nèi)部函數(shù)可以訪問(wèn)外部函數(shù)的參數(shù)和變量,內(nèi)部函數(shù)在外部函數(shù)之外被調(diào)用時(shí)锈玉,就形成了閉包爪飘。閉包是一個(gè)環(huán)境,能讀取到其他函數(shù)內(nèi)部的變量拉背。
閉包是一個(gè)函數(shù)在創(chuàng)建時(shí)允許自身函數(shù)訪問(wèn)并操作該自身函數(shù)之外的變量時(shí)所創(chuàng)建的作用域悦施。換句話說(shuō),閉包可以讓函數(shù)訪問(wèn)所有的變量和函數(shù)去团,只要這些變量和函數(shù)存在于該函數(shù)聲明時(shí)的作用域內(nèi)就行。
——《JavaScript忍者秘籍》86
當(dāng)在函數(shù)內(nèi)部定義了其他函數(shù)時(shí),就創(chuàng)建了閉包土陪。閉包有權(quán)訪問(wèn)包含函數(shù)內(nèi)部的所有變量昼汗。其原理是:在后臺(tái)執(zhí)行環(huán)境中,閉包的作用域鏈包含著它自己的作用域鬼雀、包含函數(shù)的作用域和全局作用域顷窒。通常,函數(shù)的作用域及其所有變量都會(huì)在函數(shù)執(zhí)行結(jié)束后被銷(xiāo)毀源哩。但是鞋吉,當(dāng)函數(shù)返回了一個(gè)閉包時(shí),這個(gè)函數(shù)的作用域?qū)?huì)一直在內(nèi)存中保存到閉包不存在為止励烦。
——《JavaScript高級(jí)程序設(shè)計(jì)》p192
.
閉包的優(yōu)點(diǎn):
可以在外部讀取到函數(shù)內(nèi)部的變量谓着,可以讓這些變量的值始終保持在內(nèi)存中。
避免全局變量的污染坛掠。定義過(guò)多全局變量可能會(huì)造成命名沖突赊锚,而使用閉包,外層函數(shù)的變量或?qū)傩蕴胨ǎ挥型ㄟ^(guò)內(nèi)部函數(shù)訪問(wèn)到舷蒲,無(wú)法通過(guò)其他途徑訪問(wèn)修改,從而達(dá)到了保護(hù)變量安全的效果(避免變量污染)友多。
通過(guò)閉包實(shí)現(xiàn)了 JavaScript 私有屬性和私有方法的效果牲平。若每個(gè)模塊都可以調(diào)用修改變量,當(dāng)程序越來(lái)越復(fù)雜之后域滥,會(huì)帶不可預(yù)測(cè)的危險(xiǎn)纵柿。所以推薦變量盡量私有化,當(dāng)需要讓局部變量發(fā)揮全局變量的作用時(shí)骗绕,可以考慮使用閉包藐窄。
使用閉包可以在 JavaScript 中模仿塊級(jí)作用域(JavaScript 本身沒(méi)有塊級(jí)作用域的概念),要點(diǎn)如下:創(chuàng)建并立即調(diào)用一個(gè)函數(shù)酬土,這樣既可以執(zhí)行其中的代碼荆忍,又不會(huì)在內(nèi)存中留下對(duì)該函數(shù)的引用。結(jié)果就是函數(shù)內(nèi)部的所有變量都會(huì)被立即銷(xiāo)毀——除非將某些變量賦值給了包含作用域(即外部作用域)中的變量撤缴。
閉包還可以用于在對(duì)象中創(chuàng)建私有變量刹枉,相關(guān)概念和要點(diǎn)如下。即使 JavaScript 中沒(méi)有正式的私有對(duì)象屬性的概念屈呕,但可以使用閉包來(lái)實(shí)現(xiàn)公有方法微宝,而通過(guò)公有方法可以訪問(wèn)在包含作用域中定義的變量。有權(quán)訪問(wèn)私有變量的公有方法叫做特權(quán)方法虎眨◇恚可以使用構(gòu)造函數(shù)模式镶摘、原型模式來(lái)實(shí)現(xiàn)自定義類型的特權(quán)方法,也可以使用模塊方式岳守、增強(qiáng)的模式來(lái)實(shí)現(xiàn)單例的特權(quán)方法凄敢。
因?yàn)閯?chuàng)建閉包必須維護(hù)額外的作用域,所以過(guò)度使用它們可能會(huì)占用大量?jī)?nèi)存湿痢。
由于閉包會(huì)攜帶包含它的函數(shù)的作用域涝缝,因此會(huì)比其他函數(shù)占用更多的內(nèi)存。
——《JavaScript高級(jí)程序設(shè)計(jì)》p192
.
閉包的缺點(diǎn):
閉包會(huì)使函數(shù)中的變量一直保存在內(nèi)存中譬重,不會(huì)被瀏覽器內(nèi)存回收機(jī)制清理拒逮,內(nèi)存消耗大。解決辦法是臀规,在退出函數(shù)之前滩援,將不需要使用的局部變量刪除掉。
一旦數(shù)據(jù)不再有用以现,最好通過(guò)將其值設(shè)置為
null
狠怨,來(lái)釋放其引用——這個(gè)做法叫做解除引用。適用于大多數(shù)全局變量和全局變量的屬性邑遏。局部變量會(huì)在它們離開(kāi)執(zhí)行環(huán)境時(shí)自動(dòng)被解除引用佣赖。解除引用的真正作用是讓值脫離執(zhí)行環(huán)境,以便垃圾收集器下次運(yùn)行時(shí)將其回收记盒。——《JavaScript高級(jí)程序設(shè)計(jì)》p81