首先聲明,這是一篇面向小白的博客,不過也歡迎各位大牛批評指正,謝謝。segmentfault
其實關於閉包各個論壇社區裏都有不少的文章來說它,畢竟閉包是JavaScript中一個特點,也正由於這個雨中不一樣的特點也讓閉包理解起來有一些吃力。筆者在這裏不只僅是想介紹閉包,也向列舉一些筆者所見過的一些閉包,若是有讀者還有一些比較經典的閉包例子,但願能夠在評論區裏留一下,謝謝。閉包
說了半天,究竟什麼是閉包呢?app
閉包就是函數的局部變量集合,只是這些局部變量在函數返回後會繼續存在。函數
閉包就是就是函數的「堆棧」在函數返回後並不釋放,咱們也能夠理解爲這些函數堆棧並不在棧上分配而是在堆上分配。性能
當在一個函數內定義另一個函數就會產生閉包。this
爲了便於理解,咱們能夠簡單的將閉包理解爲:code
閉包:是指有權訪問另一個函數做用域中的變量的函數。對象
JavaScript中是沒有塊級做用域的。不過關於塊級做用域咱們在這裏不作深刻探究,筆者在http://segmentfault.com/a/1190000004092842M中有對塊級做用域較爲詳細的解釋,不懂的讀者能夠去看看。ip
變量的做用域無非就是兩種:全局變量和局部變量。
Javascript語言的特殊之處,就在於函數內部能夠直接讀取全局變量。內存
var n=999; function f1(){ alert(n); } f1(); // 999
如上函數,f1可調用全局變量n
另外一方面,在函數外部天然沒法讀取函數內的局部變量。
function f1(){ var n=999; } alert(n); // error
這裏有一個地方須要注意,函數內部聲明變量的時候,必定要使用var命令。若是不用的話,你實際上聲明瞭一個全局變量。
function f1(){ n=999; } f1(); alert(n); // 999
1. 理解閉包
咱們已經理解了什麼是做用域,什麼是塊級做用域,那又該如何去訪問函數內部的變量呢?
出於種種緣由,咱們有時候須要獲得函數內的局部變量。可是,前面已經說過了,正常狀況下,這是辦不到的,只有經過變通方法才能實現。
function f1(){ var n=999; function f2(){ alert(n); } return f2; } var result=f1(); result();// 彈出999
上面函數中的f2函數就是閉包,就是經過創建函數來訪問函數內部的局部變量。
2. 閉包的用途
閉包能夠用在許多地方。它的最大用處有兩個,一個是前面提到的能夠讀取函數內部的變量,另外一個就是讓這些變量的值始終保持在內存中。
function f1(){ var n=999; nAdd=function(){n+=1} function f2(){ alert(n); } return f2; } var result=f1(); result(); // 999 nAdd(); 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,能夠在函數外部對函數內部的局部變量進行操做。
3. 閉包的注意點
1)因爲閉包會使得函數中的變量都被保存在內存中,內存消耗很大,因此不能濫用閉包,不然會形成網頁的性能問題,在IE中可能致使內存泄露。解決方法是,在退出函數以前,將不使用的局部變量所有刪除。
2)閉包會在父函數外部,改變父函數內部變量的值。因此,若是你把父函數看成對象(object)使用,把閉包看成它的公用方法(Public Method),把內部變量看成它的私有屬性(private value),這時必定要當心,不要隨便改變父函數內部變量的值。
4. 經典閉包小案例
若是你能理解下面所有的案例,那你的閉包就算是真正掌握了。
var name = "The Window"; var object = { name : "My Object", getNameFunc : function(){ return function(){ return this.name; }; } }; alert(object.getNameFunc()());//The Window
var name = "The Window"; var object = { name : "My Object", getNameFunc : function(){ var that = this; return function(){ return that.name; }; } }; alert(object.getNameFunc()());//My Object
function fun(n,o) { console.log(o) return { fun:function(m){ return fun(m,n); } }; } var a = fun(0); a.fun(1); a.fun(2); a.fun(3);//undefined,?,?,? var b = fun(0).fun(1).fun(2).fun(3);//undefined,?,?,? var c = fun(0).fun(1); c.fun(2); c.fun(3);//undefined,?,?,?
//問:三行a,b,c的輸出分別是什麼?
這是一道很是典型的JS閉包問題。其中嵌套了三層fun函數,搞清楚每層fun的函數是那個fun函數尤其重要。
//答案:
//a: undefined,0,0,0
//b: undefined,0,1,2
//c: undefined,0,1,1
都答對了麼?若是都答對了恭喜你在js閉包問題當中幾乎沒什麼能夠難住你了。