閉包理解

各類專業文獻上的「閉包」(closure)定義很是抽象,很難看懂。閉包

因爲在Javascript語言中,只有函數內部的子函數才能讀取局部變量,所以能夠把閉包簡單理解成「定義在一個函數內部的函數」。函數

因此,在本質上,閉包就是將函數內部和函數外部鏈接起來的一座橋樑。性能

 

做用域對象

要理解閉包,首先必須理解Javascript特殊的變量做用域。ip

變量的做用域無非就是兩種:全局變量和局部變量。內存

Javascript語言的特殊之處,就在於函數內部能夠直接讀取全局變量。作用域

Js代碼io

  var n=10;function

  function fn1(){
    alert(n);
  }變量

  fn1(); // 10

另外一方面,在函數外部天然沒法讀取函數內的局部變量。

Js代碼

  function fn1(){
    var n=10;
  }

  alert(n); // error

這裏有一個地方須要注意,函數內部聲明變量的時候,必定要使用var命令。若是不用的話,你實際上聲明瞭一個全局變量!

Js代碼

  function fn1(){
    n=10;
  }

  fn1();

  alert(n); // 10

 

如何從外部讀取局部變量?

出於種種緣由,咱們有時候須要獲得函數內的局部變量。可是,前面已經說過了,正常狀況下,這是辦不到的,只有經過變通方法才能實現。

那就是在函數的內部,再定義一個函數。

Js代碼

  function fn1(){

    n=10;

    function fn2(){
      alert(n); // 10
    }

  }

在上面的代碼中,函數fn2就被包括在函數fn1內部,這時fn1內部的全部局部變量,對fn2都是可見的。可是反過來就不行,fn2內部的局部變量,對fn1 就是不可見的。這就是Javascript語言特有的「鏈式做用域」結構(chain scope),

子對象會一級一級地向上尋找全部父對象的變量。因此,父對象的全部變量,對子對象都是可見的,反之則不成立。

既然fn2能夠讀取fn1中的局部變量,那麼只要把fn2做爲返回值,咱們不就能夠在fn1外部讀取它的內部變量了嗎!


Js代碼

  function fn1(){

    n=10;

    function fn2(){
      alert(n);
    }

    return fn2;

  }

  var result=fn1();

  result(); // 10

 

使用閉包的注意點

因爲閉包會使得函數中的變量都被保存在內存中,內存消耗很大,因此不能濫用閉包,不然會形成網頁的性能問題,在IE中可能致使內存泄露。

相關文章
相關標籤/搜索