我所理解的js閉包

舉個例子:閉包

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),這時必定要當心,不要隨便

改變父函數內部變量的值。

相關文章
相關標籤/搜索