• 闭包(Closure)基础分析


    闭包(Closure)

    本文聚焦于回答2个问题:

    1. 在全局作用域中,如何读取函数内部的局部变量?
    2. 在全局作用域中,如何修改函数内部的局部变量?

    变量作用域

    JavaScript语言的作用域,一句话概括就是:内层函数可以访问外层函数的变量,而外层函
    数不可以访问内层函数的变量。

    在内层函数中定义的变量如果没使用var关键词,则该变量变为全局变量。通过这种方法定义
    的全局变量,要在此函数执行后才有效。请看下面代码:

    function outer() {
        n = 820;
    
        function inner() {
            he = 835;
        }
    
        return inner;
    }
    
    // console.log(n); 
    // n is not defined
    // 上面的错误会打断程序执行,如要测试下面的代码,需注释掉上面的代码
    
    outer();
    console.log(n); // 820
    
    // console.log(he);
    // n is not defined
    
    (outer())();
    console.log(he); // 835
    

    如何从外部读取局部变量

    通过闭包可以实现在全局作用域中访问函数内部变量。

    function outer() {
        var n = 820;
    
        function inner() {
            return n;
        }
    
        return inner;
    }
    
    var k = (outer())();
    console.log(k); // 820
    

    outer()函数执行一次,将返回inner()函数的引用,再执行一次inner()
    函数,就成功的把局部变量n返回出来。从而实现从外部读取内部变量。

    如何从外部修改局部变量

    通过闭包可以实现在全局作用域中修改函数内部变量。

    var n = "hello";
    
    function outer() {
        var n = 820;
    
        function get() {
            return n;
        }
    
        function inc() {
            n++;
        }
    
        return {
            n: n,
            get: get,
            inc: inc
        };
    }
    
    var result = outer();
    
    console.log( result.n ); // 820
    console.log( result.get() ); // 820
    
    result.inc();
    console.log( result.get() ); // 821
    result.inc();
    console.log( result.get() ); // 822
    
    console.log( result.n ); // 820
    

    outer()函数返回一个对象,这个对象有1个属性,2个方法。正常情况下一个
    函数调用完毕,其内部的变量将会被垃圾回收机制(garbage collection)回收。
    也就是说这些变量已经不存在内存中,也没有办法读取这些变量的值,更没法修改。

    但是,我们把outer()函数的返回值赋给了一个全局变量,全局变量是始终存在
    内存中的,而这个全局变量result又使用到了outer()函数的局部变量,所以
    outer()函数的局部变量,不会被清除,将一直保存在内存中。

    因此,只要我们执行一次result.inc(), outer()函数里面的n就会增加1。
    而我们通过result.get()就可以访问到outer()函数里面的n

    要注意result.n的值始终是820,这里面的820只是n变量的一个副本,一旦
    outer()函数执行完毕,result.n就和n没有关系了。

    注意事项

    由于闭包会使得函数中的变量都被保存在内存中,内存消耗很大,所以不能滥用闭包
    ,否则会造成网页的性能问题。

    参考资料

    http://www.ruanyifeng.com/blog/2009/08/learning_javascript_closures.html

  • 相关阅读:
    LeetCode题目:Gray Code
    LeetCode题目: Remove Duplicate Letters
    非阻塞socket中read、write返回值
    C连接MySql
    使用GDB调试程序
    C语言中使用库函数解析命令行参数
    把服务器当网盘玩 教你从云服务器下载自己的文件
    微信小程序 地图选点 获取用户选择的定位信息 wx.chooseLocation
    如何避免高不成低不就? 疫情当下Java学习路线分享
    微信小程序 获取手机号 JS
  • 原文地址:https://www.cnblogs.com/asheng2016/p/7277097.html
Copyright © 2020-2023  润新知