在Javascript中閉包(Closure)
什么是閉包
“官方”的解釋是:所謂“閉包”,指的是一個擁有許多變量和綁定了這些變量的環(huán)境的表達式(通常是一個函數(shù)),因而這些變量也是該表達式的一部分。
相信很少有人能直接看懂這句話,因為他描述的太學術(shù)。我想用如何在Javascript中創(chuàng)建一個閉包來告訴你什么是閉包,因為跳過閉包的創(chuàng)建過程直接理解閉包的定義是非常困難的??聪旅孢@段
代碼
特點這段代碼有兩個特點:
1、函數(shù)b嵌套在函數(shù)a內(nèi)部;
2、函數(shù)a返回函數(shù)b。
這樣在執(zhí)行完var c=a( )后,變量c實際上是指向了函數(shù)b,再執(zhí)行c( )后就會彈出一個窗口顯示i的值(第一次為1)。這段代碼其實就創(chuàng)建了一個閉包,為什么?因為函數(shù)a外的變量c引用了函數(shù)a內(nèi)的函數(shù)b,就是說:
當函數(shù)a的內(nèi)部函數(shù)b被函數(shù)a外的一個變量引用的時候,就創(chuàng)建了一個閉包。
作用簡而言之,閉包的作用就是在a執(zhí)行完并返回后,閉包使得Javascript的垃圾回收機制不會收回a所占用的資源,因為a的內(nèi)部函數(shù)b的執(zhí)行需要依賴a中的變量。這是對閉包作用的非常直白的描述,不專業(yè)也不嚴謹,但大概意思就是這樣,理解閉包需要循序漸進的過程。
在上面的例子中,由于閉包的存在使得函數(shù)a返回后,a中的i始終存在,這樣每次執(zhí)行c(),i都是自加1后alert出i的值。
那 么我們來想象另一種情況,如果a返回的不是函數(shù)b,情況就完全不同了。因為a執(zhí)行完后,b沒有被返回給a的外界,只是被a所引用,而此時a也只會被b引 用,因此函數(shù)a和b互相引用但又不被外界打擾(被外界引用),函數(shù)a和b就會被回收。(關(guān)于Javascript的垃圾回收機制將在后面詳細介紹)
另一個例子模擬私有變量
function Counter(start){
var count = start;
return{ increment:function(){ count++; }, get:function(){ return count; } } } var foo =Counter(4); foo.increment(); foo.get();// 5 結(jié)果這里,Counter 函數(shù)返回兩個閉包,函數(shù) increment 和函數(shù) get。 這兩個函數(shù)都維持著 對外部作用域 Counter 的引用,因此總可以訪問此作用域內(nèi)定義的變量 count. |
|
來自: 一本正經(jīng)地胡鬧 > 《待分類》