Javascript闭包

闭包(closure)的特点就是闭包函数返回时,该函数内部变量处于激活状态,函数所在栈区依然保留。

我们所熟知的主流语言,C#/java等,在函数内部只要执行了return,函数就会返回结果。

然后内存中删除该函数所在的区域,生命周期也就停止了,一般的js函数也是这样。但是有闭包特性的js函数有点特殊。
举例子来说:

function a() {
    var i = 0;
    function b() {
        alert(++i);
    }
    return b;
}
var c = a();
c();//0
c();//1
这是个标准的闭包。在函数a中定义了函数b,a又return了b的值。这些可以先不管。

var c = a();

c();

这两句执行很重要。

在var c = a();这行里,执行了a函数,那么肯定a经过了return。

按照主流语言的函数特性,现在c的值就是a的返回值。第二行c()的执行实际执行的就是b函数。

最后不管执行的是谁,会弹出一个值为0的窗口,到此为止,所有的生命周期按理论来说就算全部结束了。

可是,c();第一次弹出0,第二次执行却弹出了1。也就是说,第一次c()后,a中的i依然保留。自然a在内存的栈区依然保留。a是return过了,但是a及内部值却依然存在,这就是闭包。

闭包是指在 JavaScript 中,内部函数b()总是可以访问其所在的外部函数a()声明的参数变量,即使在其外部函数被返回(寿命终结)了之后。
总结下闭包的特征
1,闭包外层是个函数a()
2,闭包内部都有函数b()
3,闭包会return内部函数b()
4,执行闭包后,闭包内部变量i会存在(未销毁)。

闭包的应用场景:
1、保护函数内的变量安全。以最开始的例子为例,函数a中i只有函数b才能访问,而无法通过其他途径访问到,因此保护了i的安全性。
2、在内存中维持一个变量。依然如前例,由于闭包,函数a中i的一直存在于内存中,因此每次执行c(),都会给i自加1。

很显然,没有任何实现匿名函数不可能应用了闭包特性。但如果匿名函数里面有实现,那也还得确定它的实现中有没有用到那些未销毁的内部变量。



原文地址:https://www.cnblogs.com/chrisghb8812/p/5651091.html