首先声明,这是一篇面向小白的博客,不过也欢迎各位大牛批评指正,谢谢。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闭包问题当中几乎没什么能够难住你了。