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!

相关文章
相关标签/搜索