闭包的定义:当函数可以记住并且访问所在的词法作用域的时候,就会产生闭包,即使函数是在当前词法作用域之外执行的。
function foo (){
var a = 2;
function bar(){
console.log(a); // 2
}
bar();
}
foo();
上面的代码会在控制台打出2,上面的执行结果是基于词法作用域的查找规则,函数bar()可以访问外部变量中的a的值。
上面的代码我们从逻辑上看,可以看出,这个并不是完全的闭包,因为bar()只是对a的引用方法是词法作用域的查找规则,而这些规则只是闭包定义中的一部分,所以我们再看一下下面的一段的代码:
function foo (){
var a = 2;
function bar (){
console.log(a)
}
return bar;
}
var baz = foo();
baz(); // 2
函数bar()的词法作用域可以访问foo()的内部作用域。然后我们将bar()函数本身当做一个值类型进行传递,在这个例子中,我们将bar所引用函数本身当做一个值类型进行传递,在这个例子中,我们将bar所引用的函数本身当做返回值。
在foo执行之后,将其返回值,也就是bar的内部函数直接返回给了baz变量,并且调用baz(),实际上只是通过不同的标识符引用调用了内部的函数bar();
我们再foo()执行之后,通常会期待foo()的整个内部作用域都会被销销毁,因为我们的引擎都是拥有垃圾回收机制的,这个机制会及时的释放内存,因为一般情况下,执行过的foo,引擎一般会认为是没有必要再次继续执行的,所以会对这个部分的内纯进行垃圾回收。然而“闭包”就是一个例外的场景,事实上内部的作用域是依然存在的,内部的内纯并没有被回收,那么是谁在使用这个内存呢?其实,就是bar()本身在使用这个部分的内存。
因为bar()的声明位置,它拥有涵盖foo()内部作用域的闭包,让转让工人作用域内可以一直存在,以供bar()在任何时候进行调用。bar()依然持有对该作用域的引用,而这个引用就叫做闭包。
网友评论