1.什么是闭包?
《JavaScript高级程序设计》这样描述:
闭包是指有权访问另一个函数作用域中的变量的函数;
《JavaScript权威指南》这样描述:
函数对象可以通过作用域链相互关联起来,函数体内部的变量都可以保存在函数作用域中,这种特性在计算机科学文献中被称为‘闭包’
从技术的角度讲,所有的JavaScript函数都是闭包:它们都是对象,它们都关联到作用域链。
MDN中这样说:
闭包是函数和声明该函数的词法环境的组合
《你不知道的JavaScript》这样描述:
当函数可以记住并访问所在的词法作用域时,就产生了闭包,即使函数是在当前词法作用域之外执行。
最后一种说法在我浏览各种关于闭包的理解时,出现的最为频繁,也较为易懂。
词法作用域是由函数声明时书写代码的位置决定的,而闭包是词法作用域形成的自然结果。当在函数内部声明了内部函数,并将内部函数作为值返回,就会产生闭包。
function fn1() {
var name = 'Tilo';
function fn2() {
console.log(name);
}
return fn2;
}
var fn3 = fn1();
fn3();
注:在这个代码中,我们如果直接在全局作用域下调用fn2函数,就会报错ReferenceError,这是跟作用域判断有关,因为引擎首先会根据词法作用域的查询规则RHS。
在作用域中查找变量都是RHS,并且查找的规则是从当前作用域开始找,如果没找到再到父级作用域中找,一层层往外找,如果在全局作用域如果还没找到的话,就会报错了:ReferenceError: 某变量 is not defined。
将内部函数作为值传递并且return,在调用了fn1函数之后,fn2函数会保持对fn1的词法作用域的引用,记住并可以访问所在的词法作用域(这里就是fn1的作用域)。看到一位老师的比喻,讲说:将函数作为值传递,实际就是打开了一条访问内部变量的通道。这‘通道’说的就是内部函数对词法作用域的引用,这个引用就是闭包。
换言之,当fn1函数执行完毕之后,其作用域是会被销毁的,然后垃圾回收器会释放那段内存空间。而闭包却很神奇的将fn1的作用域存活下来,fn2依然持有该作用域的引用,这个引用就是闭包。
总结:某个函数在定义时的词法作用域之外的地方被调用,闭包可以使该函数继续访问定义时的词法作用域。
2.闭包的特点
①基于词法作用域的查找规则。
②在一个函数内部定义一个内部函数,然后将内部函数作为值返回,或者直接或者间接的立即执行内部函数。
③拥有更长的生命周期,保持对当前词法作用域的引用。
❶函数作为值返回
function fn1() {
var num = 23;
function fn2() {
return num;
}
return fn2;
}
var fn3 = fn1();
console.log(fn3()); //23
正常情况下,fn1函数执行之后,在JS中fn1函数的作用域就被销毁,内存空间被释放,因为这里将函数作为值返回了,产生了闭包,因为闭包的作用,我们仍然可以在局部作用域里访问到num,所以fn1的的词法作用域仍然存在,证明了闭包可以访问所在的词法作用域,且拥有更长的生命周期,保持对当前词法作用域的引用。
❷直接或间接的执行内部函数
function fn1(){
var num = 20;
function fn2(){
console.log(num)
}
fn2();
}
fn1() ; //20
上面代码中,一样形成了闭包。并且产生的结果与第一种情况相同。
参考凯斯老师在什么是闭包?中的回答,侵删。