JS(七)闭包

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 上面代码中,一样形成了闭包。并且产生的结果与第一种情况相同。 参考凯斯老师在什么是闭包?中的回答,侵删。

本文章由javascript技术分享原创和收集

发表评论 (审核通过后显示评论):