JS(七)闭包

作者: 长鲸向南 | 来源:发表于2020-01-04 18:11 被阅读0次

    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
    

    上面代码中,一样形成了闭包。并且产生的结果与第一种情况相同。

    参考凯斯老师在什么是闭包?中的回答,侵删。

    相关文章

      网友评论

        本文标题:JS(七)闭包

        本文链接:https://www.haomeiwen.com/subject/zcrfactx.html