zoukankan      html  css  js  c++  java
  • 狗日的js的闭包

    一、变量的作用域
    要懂得闭包,起首必须懂得Javascript特别的变量作用域。
    变量的作用域无非就是两种:全局变量和局部变量。
    Javascript说话的特别之处,就在于函数内部可以直接读取全局变量。

    Js代码
      var n=999;
      function f1(){
        alert(n);
      }
      f1(); // 999
    另一方面,在函数外部天然无法读取函数内的局部变量。
    Js代码
      function f1(){
        var n=999;
      }
      alert(n); // error
    这里有一个处所须要重视,函数内部声明变量的时候,必然要用var。若是不用的话,你实际上声明了一个全局变量!
    Js代码
      function f1(){
        n=999;
      }
      f1();
      alert(n); // 999

    二、如何从外部读取局部变量?
    出于各种原因,我们有时要获得函数内的局部变量。然则,前面已经说过了,正常情况下,这是办不到的,只有经由过程变通才能实现。
    那就是在函数的内部,再定义一个函数。
    Js代码
      function f1(){
        n=999;
        function f2(){
          alert(n); // 999
        }
      }
    在上方的代码中,函数f2就被包含在函数f1内部,这时f1内部的所有局部变量,对f2都是可见的。然则反过来就不可,f2内部的局部变量,对f1 就是不成见的。这就是Javascript说话特有的“链式作用域”布局(chain scope),
    子对象会一级一级地向上寻找所有父对象的变量。所以,父对象的所有变量,对子对象都是可见的,反之则不成立。
    既然f2可以读取f1中的局部变量,那么只要把f2作为返回值,我们不就可以在f1外部读取它的内部变量了吗!

    Js代码
      function f1(){
        n=999;
        function f2(){
          alert(n);
        }
        return f2;
      }
      var result=f1();
      result(); // 999

    三、闭包的概念
    上一节代码中的f2函数,就是闭包。
    各类专业文献上的“闭包”(closure)定义很是抽象,很丢脸懂。我的懂得是,闭包就是可以或许读取其他函数内部变量的函数。
    因为在Javascript说话中,只有函数内部的子函数才干读取局部变量,是以可以把闭包简单懂得成“定义在一个函数内部的函数”。
    所以,在本质上,闭包就是将函数内部和函数外部连接起来的一座桥梁。
    --------------------------------------------------------------------------------------------------------b
    四、闭包的用处
    闭包可以用在很多处所。它的最大用处有两个,一个是前面提到的可以读取函数内部的变量,另一个就是让这些变量的值始终对峙在内存中。
    怎么来懂得这句话呢?请看下面的代码。

    Js代码
      function f1(){
        var n=999;
        nAdd=function(){n+=1}
        function f2(){
          alert(n);
        }
        return f2;
      }
      var result=f1();
      result(); // 999
      nAdd();
      result(); // 1000
    在这段代码中,result实际上就是闭包f2函数。它一共运行了两次,第一次的值是999,第二次的值是1000。这就说明,函数f1中的局部变量n一向保存在内存中,并没有在f1调用后被主动清除。
    为什么会这样呢?原因就在于f1是f2的父函数,而f2被赋给了一个全局变量,这导致f2始终在内存中,而f2的存在依附于f1,是以f1也始终在内存中,不会在调用停止后,被垃圾收受接管机制(garbage collection)收受接管。
    这段代码中另一个值得重视的一处,就是“nAdd=function(){n+=1}”这一行,起首在nAdd前面没有应用var关键字,是以 nAdd是一个全局变量,而不是局部变量。其次,nAdd的值是一个匿名函数(anonymous function),而这个
    匿名函数本身也是一个闭包,所以nAdd相当于是一个setter,可以在函数外部对函数内部的局部变量进行操纵。

    五、应用闭包的重视点
    1)因为闭包会使得函数中的变量都被保存在内存中,内存消费很大,所以不要滥用闭包,不然会造成网页的性能题目,在IE中可能导致内存泄漏。解决办法是,在退出函数之前,将不应用的局部变量全部删除。
    2)闭包会在父函数外部,改变父函数内部变量的值。所以,若是你把父函数当做难象(object)应用,把闭包算作它的公用办法(Public Method),把内部变量算作它的私有属性(private value),这时必然要警惕,不要随便
    改变父函数内部变量的值。

    有权访问另一个函数作用域内变量的函数都是闭包。

    什么是闭包?

    先看一段代码:

    function a(){
    var n = 0;
    function inc() {
    n++;
    console.log(n);
    }
    inc();
    inc();
    }
    a(); //控制台输出1,再输出2
    简单吧。再来看一段代码:

    function a(){
    var n = 0;
    this.inc = function () {
    n++;
    console.log(n);
    };
    }
    var c = new a();
    c.inc(); //控制台输出1
    c.inc(); //控制台输出2
    简单吧。

    什么是闭包?这就是闭包!

    有权访问另一个函数作用域内变量的函数都是闭包。

    这里 inc 函数访问了构造函数 a 里面的变量 n,所以形成了一个闭包。

    再来看一段代码:

    function a(){
    var n = 0;
    function inc(){
    n++;
    console.log(n);
    }
    return inc;
    }
    var c = a();
    c(); //控制台输出1
    c(); //控制台输出2
    看看是怎么执行的:

    var c = couter(),这一句 couter()返回的是函数 inc,那这句等同于 var c = inc;

    c(),这一句等同于 inc(); 注意,函数名只是一个标识(指向函数的指针),而()才是执行函数。

    后面三句翻译过来就是: var c = inc; inc(); inc();,跟第一段代码有区别吗? 没有。

    什么是闭包?这就是闭包!

    所有的教科书教程上都喜欢用最后一段来说明闭包,但我觉得这将问题复杂化了。这里面返回的是函数名,没看过谭浩强C/C++程序设计的同学可能一下子没反应出带不带()的区别,也就是说这种写法自带一个陷阱。虽然这种写法更显高大上,但我还是喜欢将问题单一化,看看代码 1 和代码 2,你还会纠结函数的调用,你会纠结 n 的值吗?

    为啥要这样写?

    我们知道,js的每个函数都是一个个小黑屋,它可以获取外界信息,但是外界却无法直接看到里面的内容。将变量 n 放进小黑屋里,除了 inc 函数之外,没有其他办法能接触到变量 n,而且在函数 a 外定义同名的变量 n 也是互不影响的,这就是所谓的增强“封装性”。

    而之所以要用 return 返回函数标识 inc,是因为在 a 函数外部无法直接调用 inc 函数,所以 return inc 与外部联系起来,代码 2 中的 this 也是将 inc 与外部联系起来而已。

    常见的陷阱

    看看这个:

    function createFunctions(){
    var result = new Array();
    for (var i=0; i < 10; i++){
    result[i] = function(){
    return i;
    };
    }
    return result;
    }
    var funcs = createFunctions();
    for (var i=0; i < funcs.length; i++){
    console.log(funcsi);
    }
    乍一看,以为输出 0~9 ,万万没想到输出10个10?

    这里的陷阱就是:函数带()才是执行函数! 单纯的一句 var f = function() { alert(‘Hi’); }; 是不会弹窗的,后面接一句 f(); 才会执行函数内部的代码。上面代码翻译一下就是:

    var result = new Array(), i;
    result[0] = function(){ return i; }; //没执行函数,函数内部不变,不能将函数内的i替换!
    result[1] = function(){ return i; }; //没执行函数,函数内部不变,不能将函数内的i替换!
    ...
    result[9] = function(){ return i; }; //没执行函数,函数内部不变,不能将函数内的i替换!
    i = 10;
    funcs = result;
    result = null;

    console.log(i); // funcs0就是执行 return i 语句,就是返回10
    console.log(i); // funcs1就是执行 return i 语句,就是返回10
    ...
    console.log(i); // funcs9就是执行 return i 语句,就是返回10
    为什么只垃圾回收了 result,但却不收了 i 呢? 因为 i 还在被 function 引用着啊。好比一个餐厅,盘子总是有限的,所以服务员会去巡台回收空盘子,但还装着菜的盘子他怎么敢收? 当然,你自己手动倒掉了盘子里面的菜(=null),那盘子就会被收走了,这就是所谓的内存回收机制。

    至于 i 的值怎么还能保留,其实从文章开头一路读下来,这应该没有什么可以纠结的地方。盘子里面的菜,吃了一块不就应该少一块吗?

    总结一下

    闭包就是一个函数引用另外一个函数的变量,因为变量被引用着所以不会被回收,因此可以用来封装一个私有变量。这是优点也是缺点,不必要的闭包只会徒增内存消耗!另外使用闭包也要注意变量的值是否符合你的要求,因为他就像一个静态私有变量一样。闭包通常会跟很多东西混搭起来,接触多了才能加深理解,这里只是开个头说说基础性的东西。

  • 相关阅读:
    邀请函|2021 云原生实战峰会,邀请您免费现场参会报名
    Game On Serverless:SAE 助力广州小迈提升微服务研发效能
    说出你和「云原生」的故事,获得年度云原生顶级盛会通行证
    巧用浏览器F12调试器定位系统前后端bug
    测试人员怎样定位bug原因
    mysql删除某个表前100条数据
    设计模式之工厂方法模式
    2021.11.19
    20211117
    JQuery插件集合
  • 原文地址:https://www.cnblogs.com/libin-1/p/5745648.html
Copyright © 2011-2022 走看看