JavaScript閉包初探

JavaScript的閉包

首先聲明,這是一篇面向小白的博客,不過也歡迎各位大牛批評指正,謝謝。segmentfault

  其實關於閉包各個論壇社區裏都有不少的文章來說它,畢竟閉包是JavaScript中一個特點,也正由於這個雨中不一樣的特點也讓閉包理解起來有一些吃力。筆者在這裏不只僅是想介紹閉包,也向列舉一些筆者所見過的一些閉包,若是有讀者還有一些比較經典的閉包例子,但願能夠在評論區裏留一下,謝謝。閉包

說了半天,究竟什麼是閉包呢?app

  • 閉包就是函數的局部變量集合,只是這些局部變量在函數返回後會繼續存在。函數

  • 閉包就是就是函數的「堆棧」在函數返回後並不釋放,咱們也能夠理解爲這些函數堆棧並不在棧上分配而是在堆上分配。性能

  • 當在一個函數內定義另一個函數就會產生閉包。this

爲了便於理解,咱們能夠簡單的將閉包理解爲:code

  • 閉包:是指有權訪問另一個函數做用域中的變量的函數。對象

JavaScript中的做用域

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閉包問題當中幾乎沒什麼能夠難住你了。

Happy hacking!

相關文章
相關標籤/搜索