目录
- 引言:对于JavaScript来说,理解闭包可以看作是某种意义上的重生。JavaScript中闭包无处不在,并不是一个需要学习的新语法或模式才能使用的工具。我们总是在无意识中使用着闭包。
理解闭包
- 如下代码清晰地展示了闭包:
function foo() {
var a = 2;
function bar() {
console.log(a);
}
return bar;
}
var baz = foo();
baz(); // 2 -- 这就是闭包的效果
-
解释:
-
函数bar()的词法作用域能够访问foo()的内部作用域。我们将bar()函数本身当作一个值类型进行传递。在这个例子中,我们将bar所引用的函数对象本身当作返回值。
-
在foo()执行后,其返回值(也就是内部的bar()函数)赋值给变量baz并调用baz(),实际上只是通过不同的标识符引用调用了内部的函数bar()。
-
bar()显然可以被正常执行。但在这个例子中,它在自己定义的词法作用域以外被执行。
-
在foo()执行后,通常会期待foo()的整个内部作用域都被销毁,因为我们知道引擎有垃圾回收器用来释放不再使用的内存空间。
-
由于看上去foo()的内容不会再被使用,所以很自然地会考虑对其进行回收。而闭包的“神奇”之处正是可以组织这件事情的发生。事实上内部作用域依然存在,因此没有被回收。谁在使用这个内部作用域?bar()本身在使用。
-
拜bar()所声明的位置所赐,它拥有涵盖foo()内部作用域的闭包,使得该作用域能够一直存活,以供bar()在之后任何时间进行引用。
-
闭包是什么?:bar()依然持有对该作用域的引用,而这个引用就叫做闭包。
-
摘取自《你不知道的JavaScript》