舉個例子:閉包
1 function f1(){ 2 3 var n=999; 4 5 function f2(){ 6 alert(n); // 999 7 } 8 9 }
上面代碼中,f2()能夠讀取f1()中的局部變量n的值,可是f1()不能反過來讀取f2()中的變量,這就是js中鏈式做用域的概念。即:子級能夠逐級向上讀取上一級的內容,並且一旦讀到就再也不繼續讀取,讀不到則返回undefined。可是反過來是不成立的。函數
因此就有了閉包的概念。性能
閉包:閉包就是可以讀取其餘函數內部變量的函數。spa
因爲在Javascript語言中,只有函數內部的子函數才能讀取局部變量,所以能夠把閉包簡單理解成「定義在一個函數內部的函數」。code
因此,在本質上,閉包就是將函數內部和函數外部鏈接起來的一座橋樑。對象
1 function f1(){ 2 3 var n=999; 4 5 function f2(){ 6 alert(n); 7 } 8 9 return f2; 10 11 } 12 13 var result=f1(); 14 15 result(); // 999
閉包主要用途:blog
閉包能夠用在許多地方。它的最大用處有兩個,一個是前面提到的能夠讀取函數內部的變量,另外一個就是讓這些變量的值始終保持在內存中。ip
1 function f1(){ 2 3 var n=999; 4 5 nAdd=function(){n+=1} 6 7 function f2(){ 8 alert(n); 9 } 10 11 return f2; 12 13 } 14 15 var result=f1(); 16 17 result(); // 999 18 19 nAdd(); 20 21 result(); // 1000
在這段代碼中,result實際上就是閉包f2函數。它一共運行了兩次,第一次的值是999,第二次的值是1000。這證實了,函數f1中的局部變量n一直保存在內存中,並無在f1調用後被自動清除。內存
爲何會這樣呢?緣由就在於f1是f2的父函數,而f2被賦給了一個全局變量,這致使f2始終在內存中,而f2的存在依賴於f1,所以f1也始終在內存中,不會在調用結束後,被垃圾回收機制(garbage collection)回收。作用域
這段代碼中另外一個值得注意的地方,就是「nAdd=function(){n+=1}」這一行,首先在nAdd前面沒有使用var關鍵字,所以 nAdd是一個全局變量,而不是局部變量。其次,nAdd的值是一個匿名函數(anonymous function),而這個
匿名函數自己也是一個閉包,因此nAdd至關因而一個setter,能夠在函數外部對函數內部的局部變量進行操做。
5、使用閉包的注意點
1)因爲閉包會使得函數中的變量都被保存在內存中,內存消耗很大,因此不能濫用閉包,不然會形成網頁的性能問題,在IE中可能致使內存泄露。解決方法是,在退出函數以前,將不使用的局部變量所有刪除。
2)閉包會在父函數外部,改變父函數內部變量的值。因此,若是你把父函數看成對象(object)使用,把閉包看成它的公用方法(Public Method),把內部變量看成它的私有屬性(private value),這時必定要當心,不要隨便
改變父函數內部變量的值。