- 相關(guān)推薦
javascript的閉包概念怎么理解
接觸javascript很久了,每次理解閉包都似是而非,最近在找Web前端的工作,所以需要把基礎(chǔ)夯實(shí)一下。
本文是參照了joy_lee的博客 閉包 在她這篇博客的基礎(chǔ)上以批注的形式力爭把我的理解闡述出來,如果有不當(dāng)之處,歡迎批評(píng)指正。
《高級(jí)程序設(shè)計(jì)》上,這樣說:當(dāng)在函數(shù)內(nèi)部定義了其他函數(shù)時(shí)候,就創(chuàng)建了閉包。閉包有權(quán)訪問包含函數(shù)內(nèi)部的所有變量。
(這句話怎么理解呢?照這句話理解的話,閉包就是一個(gè)嵌套函數(shù)嘛!嵌套函數(shù)對包含它的函數(shù)的變量當(dāng)然可以訪問,這是沒有問題的。)
一般來說,內(nèi)部函數(shù)是能夠訪問到上一級(jí)乃至全局的的變量的,那么就有人這樣理解:通過閉包,可以實(shí)現(xiàn)外部訪問函數(shù)局部內(nèi)的變量。
(如果我們把作用域簡單的分個(gè)級(jí)的話,假設(shè)全局作用域作為第一級(jí),其中定義的函數(shù)體內(nèi)部作用域作為第二級(jí),在第二級(jí)作用域內(nèi)嵌套定義的函數(shù)體內(nèi)部作用域作為第三級(jí),....等等,傳統(tǒng)意義上,第一級(jí)不能訪問第二級(jí)的變量(這種變量叫做局部變量),第二級(jí)不能訪問第三級(jí),...,而反過來是可以的,這就是作用域鏈。本級(jí)作用域內(nèi)找不到再到上一級(jí)找,直至第一級(jí)全局。而閉包這種機(jī)制可以在第一級(jí)作用域中通過第三級(jí)作用域引用到第二級(jí)作用域中的變量,而方法就是在第二級(jí)作用域向第一級(jí)作用域返回?fù)碛械谌?jí)作用域的函數(shù)引用。 這個(gè)引用才是關(guān)鍵,因?yàn)檫@個(gè)引用的存在,相關(guān)的第三作用域與第二作用域都成了這個(gè)引用運(yùn)行的上下文,迫使垃圾回收機(jī)制GC不能回收這條鏈上所占用的資源。而如果沒有這個(gè)引用,則跟一般函數(shù)一樣,函數(shù)運(yùn)行完資源就會(huì)被回收。而我的疑惑也在于此,閉包單指函數(shù)中的嵌套函數(shù)還是指被第一級(jí)引用了的嵌套函數(shù)?還是都是?還是說閉包并不是嵌套函數(shù)而是嵌套函數(shù)被第一級(jí)作用域引用時(shí)所形成的這種機(jī)制?)
function a(){ var i=0; function b(){ alert(++i); } return b; } var c = a(); c();
即,閉包的作用就是在a執(zhí)行完并返回后,閉包使得Javascript的垃圾回收機(jī)制GC不會(huì)收回a所占用的資源,因?yàn)閍的內(nèi)部函數(shù)b的執(zhí)行需要依賴a中的變量。
由于閉包的存在使得函數(shù)a返回后,a中的i始終存在,這樣每次執(zhí)行c(),i都是自加1后alert出i的值。
那么,如果a不返回函數(shù)b,情況就完全不同了。因?yàn)閍執(zhí)行完后,b沒有被返回給a的外界,只是被a所引用,而此時(shí)a也只會(huì)被b引用,因此函數(shù)a和b互相引用但又不被外界打擾(被外界引用),函數(shù)a和b就會(huì)被GC回收。
實(shí)際上是就是閉包延長變量的生命周期。通常函數(shù)的作用域即變量會(huì)在函數(shù)執(zhí)行結(jié)束后被銷毀,但當(dāng)函數(shù)返回一個(gè)閉包,只要閉包不被釋放,整條作用域鏈都會(huì)占用內(nèi)存。(閉包延長變量的生命周期,這是指被第一級(jí)引用的情況。但如果沒有這個(gè)引用,閉包還能稱其為閉包嗎?)
說道作用域鏈:即 函數(shù)自己的作用域、上一層的函數(shù)的作用域....和全局作用域。訪問一個(gè)變量時(shí),自己的沒有,就一層層往上找,直至全局,若還沒有,就報(bào)錯(cuò)。
。ê芟胪虏垡痪洌]包的作用域鏈?zhǔn)菑澋。?/p>
PS: 有網(wǎng)友推薦了另一篇文章javascript閉包概念簡單解析,是不是權(quán)威暫且不提,總算有一個(gè)明確的概念。
說了這么多,閉包到底是什么,下面做一下總結(jié):
閉包是一個(gè)概念,它描述了函數(shù)執(zhí)行完畢內(nèi)存釋放后,依然內(nèi)存駐留的一個(gè)現(xiàn)象,只要把握這個(gè)核心概念,閉包就不難理解了。
【javascript的閉包概念怎么理解】相關(guān)文章:
理解Javascript對象06-21
如何理解“國學(xué)”概念04-28
常用的JavaScript模式08-29
JavaScript語法分析06-21
詳細(xì)解說JavaScript事件06-20
JavaScript常用方法匯總08-26