启示
对于那些有一点js使用经验但从未真正理解闭包概念的人说,理解闭包可以看做是某种意义上的重生。
js中闭包无处不在,闭包并不是一个需要学习新的语法或模式才能使用的工具。
闭包是基于词法作用域书写的代码时所产生的自然结果。
实质定义:当函数可以记住并访问所在的词法作用域时,就产生了闭包,即使函数是在当前词法作用域之外执行。
我们来看一段代码,清晰的展示了闭包:
function foo(){
var a=2;
function bar(){
console.log(a);
}
return bar;
}
var baz=foo();
baz();//2
函数bar()的词法作用域能够访问foo()的内部作用域,bar所引用的函数对象本身是返回值。
在foo执行后,其返回值(bar函数)赋值给变量baz并调用baz,实际上只是通过不同的标识符引用调用了内部函数bar();
bar可以被正常的执行,但是在这个例子中,他在自己定义的词法作用域以外的地方执行。
词法作用域是检索,访问,修改变量的一套规则,也可以认为是“势力范围”,根据之前的定义,bar函数所创建的词法作用域如下图方框中的内容所示:
根据本人的分析“创建”和“定义”应该不是一个含义,理由如下:
考虑一下函数的执行地点,可以理解为被调用的地点或者是函数本身执行栈的起点,个人倾向认为是函数的被调用地点,因为,如果指的是函数本身执行栈的起点,那么无论哪种情况,函数执行的地点都是相同的(无论是被直接调用还是作为返回值被调用),也就不存在“在自己定义的词法作用域以外的地方执行”。
根据文中的指示,函数在在自己定义的词法作用域以外的地方执行,这个自己定义的词法作用域应该不是函数创建的词法作用域,它也不是全局作用域,否则,该函数就在自己定义的词法作用域内执行了,因此,本人认为,函数所定义的词法作用域,应该指他的父级作用域,在上述例子中,bar函数所定义的词法作用域,就是foo函数创造的词法作用域,即该函数可以被检索并直接调用到的作用域!
在foo函数执行后,通常会期待foo的整个内部作用域都被销毁,因为我们知道引擎有垃圾回收器来释放不再使用的空间。
但是闭包能够阻止这件事情的发生,事实上内部作用域依然存在,因为bar依然在使用这个作用域。
bar依然持有该作用域的引用,而这个引用叫做闭包!