推薦阮一峰的博客
http://www.ruanyifeng.com/blog/2009/08/learning_javascript_closures.html
還有這個(gè)人的博客(強(qiáng)烈推薦里面的東西矛辕,全面詳細(xì)=)
http://www.108js.com/search.jsp
下面是自己的一些理解
<h2>首先要理解變量作用域:</h2>
- 全局變量
- 函數(shù)內(nèi)部可以直接訪問(wèn)全局變量
var n=999;
function f1(){
alert(n);
}
f1(); // 999
- 局部變量
- 在函數(shù)外部自然無(wú)法讀取函數(shù)內(nèi)的局部變量
function f1(){
var n=999;
}
alert(n); // error
了解了作用域后茂翔,思考一個(gè)問(wèn)題,如果想要在函數(shù)外部讀取局部變量怎么辦杈绸?
在函數(shù)內(nèi)部再定義一個(gè)函數(shù)就可以訪問(wèn)局部變量了。
<h2>從外部讀取局部變量</h2>
function f1(){
var n=999;
function f2(){
alert(n); // 999
}
}
在上面的代碼中矮瘟,函數(shù)f2就被包括在函數(shù)f1內(nèi)部瞳脓,這時(shí)f1內(nèi)部的所有局部變量,對(duì)f2都是可見(jiàn)的澈侠。但是反過(guò)來(lái)就不行劫侧,f2內(nèi)部的局部變量,對(duì)f1就是不可見(jiàn)的哨啃。這就是Javascript語(yǔ)言特有的<b>"鏈?zhǔn)阶饔糜?</b>結(jié)構(gòu)(chain scope)烧栋,子對(duì)象會(huì)一級(jí)一級(jí)地向上尋找所有父對(duì)象的變量。所以棘催,父對(duì)象的所有變量劲弦,對(duì)子對(duì)象都是可見(jiàn)的,反之則不成立醇坝。
既然f2可以讀取f1中的局部變量邑跪,那么只要把f2作為返回值,我們不就可以在f1外部讀取它的內(nèi)部變量了嗎呼猪!
function f1(){
var n=999;
function f2(){
alert(n);
}
return f2;
}
var result=f1();
result(); // 999
<h2>閉包</h2>
上面例子中的函數(shù)f2()就是個(gè)閉包画畅。<b>閉包就是能夠讀取其他函數(shù)內(nèi)部變量的函數(shù)</b>。
由于在Javascript語(yǔ)言中宋距,只有函數(shù)內(nèi)部的子函數(shù)才能讀取局部變量轴踱,因此可以把閉包簡(jiǎn)單理解成"定義在一個(gè)函數(shù)內(nèi)部的函數(shù)"。
所以谚赎,在本質(zhì)上淫僻,<b>閉包就是將函數(shù)內(nèi)部和函數(shù)外部連接起來(lái)的一座橋梁</b>诱篷。
<h2>閉包的用途</h2>
閉包可以用在許多地方。它的最大用處有兩個(gè)雳灵,一個(gè)是前面提到的可以讀取函數(shù)內(nèi)部的變量棕所,另一個(gè)就是讓這些變量的值始終保持在內(nèi)存中。
function f1(){
var n=999;
nAdd=function(){n+=1}
function f2(){
alert(n);
}
return f2;
}
var result=f1();
result(); // 999
nAdd();
result(); // 1000
在這段代碼中悯辙,result實(shí)際上就是閉包f2函數(shù)琳省。它一共運(yùn)行了兩次,第一次的值是999躲撰,第二次的值是1000针贬。這證明了,函數(shù)f1中的局部變量n一直保存在內(nèi)存中拢蛋,并沒(méi)有在f1調(diào)用后被自動(dòng)清除桦他。
為什么會(huì)這樣呢?原因就在于f1是f2的父函數(shù)瓤狐,而f2被賦給了一個(gè)全局變量瞬铸,這導(dǎo)致f2始終在內(nèi)存中,而f2的存在依賴于f1础锐,因此f1也始終在內(nèi)存中嗓节,不會(huì)在調(diào)用結(jié)束后,被垃圾回收機(jī)制(garbage collection)回收皆警。
這段代碼中另一個(gè)值得注意的地方拦宣,就是"nAdd=function(){n+=1}"這一行,首先在nAdd前面沒(méi)有使用var關(guān)鍵字信姓,因此nAdd是一個(gè)全局變量鸵隧,而不是局部變量。其次意推,nAdd的值是一個(gè)匿名函數(shù)(anonymous function)豆瘫,而這個(gè)匿名函數(shù)本身也是一個(gè)閉包,所以nAdd相當(dāng)于是一個(gè)setter菊值,可以在函數(shù)外部對(duì)函數(shù)內(nèi)部的局部變量進(jìn)行操作外驱。
<h2>使用閉包需要注意事項(xiàng)</h2>
1)由于閉包會(huì)使得函數(shù)中的變量都被保存在內(nèi)存中,內(nèi)存消耗很大腻窒,所以不能濫用閉包昵宇,否則會(huì)造成網(wǎng)頁(yè)的性能問(wèn)題,在IE中可能導(dǎo)致內(nèi)存泄露儿子。解決方法是瓦哎,在退出函數(shù)之前,將不使用的局部變量全部刪除。
2)閉包會(huì)在父函數(shù)外部蒋譬,改變父函數(shù)內(nèi)部變量的值割岛。所以,如果你把父函數(shù)當(dāng)作對(duì)象(object)使用羡铲,把閉包當(dāng)作它的公用方法(Public Method)蜂桶,把內(nèi)部變量當(dāng)作它的私有屬性(private value),這時(shí)一定要小心也切,不要隨便改變父函數(shù)內(nèi)部變量的值。
關(guān)于閉包的思考題:
代碼一:
var name = "The Window";
var object = {
name : "My Object",
getNameFunc : function(){
return function(){
return this.name;
};
}
};
alert(object.getNameFunc()());
這段代碼中腰湾,
getNameFunc: function() {//假設(shè)函數(shù)名為A
return function()/假設(shè)函數(shù)名為B/ { return this.name; };
}
在函數(shù)里面構(gòu)建函數(shù)的時(shí)候雷恃,閉包產(chǎn)生。
在函數(shù)B內(nèi)調(diào)用函數(shù)A的this.name,由于函數(shù)A沒(méi)有name屬性费坊,所以就去找全局變量name倒槐,找到了,所以返回“The Window”附井,要是沒(méi)有找到讨越,則返回“undefined”。
代碼二:
var name = "The Window";
var object = {
name : "My Object",
getNameFunc : function(){
var that = this;
return function(){
return that.name;
};
}
};
alert(object.getNameFunc()());
內(nèi)部函數(shù)可以訪問(wèn)定義它們的外部函數(shù)的參數(shù)和變量(除了this和arguments之外)
如果需要訪問(wèn)對(duì)象的name屬性的話永毅,就需要顯示的定義一個(gè)變量that來(lái)引用this把跨,而這個(gè)變量此時(shí)就指向object對(duì)象了。