• js闭包深度讲解


    js的闭包是学习js过程中的重点,但是不得不说也是一个难点呀,其涉及到了js中的很多概念。我在学习js中也遇到了很多问题,这篇文章算是一个对闭包的总结,文章主要内容为闭包的基本知识点与对其理解上的一些难点的讲解。

      闭包是什么?

    闭包就是嵌套在函数里面的内部函数,并且该内部函数可以访问外部函数中声明的所有局部变量、参数和其他内部函数。当该内部函数在外部函数外被调用,就生成了闭包。注意这里的调用两个字,如果内部函数没有被外部函数返回的话就不能被调用了。

     

    function outerFun()
     {
      var a=0;
      function innerFun()
      {
       a++;
       alert(a);
      }    
     }
    innerFun();

      这里的innerFun就不能被外部调用,因为其在outerFun中定义的,作用域为outerFun的范围。

    function outerFun()
    {
     var a=0;
     function innerFun()
     {
      a++;
      alert(a);
     }
     return innerFun;  //注意这里
    }

      在outerFun内部将innerFun返回到外部就能在外部调用innerFun了。

      为什么内部定义的函数只有在返回到外部后才能使用?

    这与js的运行机制有关:

    如果要更加深入的了解闭包以及函数a和嵌套函数b的关系,我们需要引入另外几个概念:函数的执行环境(excution context)、活动对象(call object)、作用域(scope)、作用域链(scope chain)。以函数a从定义到执行的过程为例阐述这几个概念。

     

    1. 定义函数a的时候,js解释器会将函数a的作用域链(scope chain)设置为定义a时a所在的“环境”,如果a是一个全局函数,则scope chain中只有window对象。
    2. 执行函数a的时候,a会进入相应的执行环境(excution context)
    3. 在创建执行环境的过程中,首先会为a添加一个scope属性,即a的作用域,其值就为第1步中的scope chain。即a.scope=a的作用域链。
    4. 然后执行环境会创建一个活动对象(call object)。活动对象也是一个拥有属性的对象,但它不具有原型而且不能通过JavaScript代码直接访问。创建完活动对象后,把活动对象添加到a的作用域链的最顶端。此时a的作用域链包含了两个对象:a的活动对象和window对象。
    5. 下一步是在活动对象上添加一个arguments属性,它保存着调用函数a时所传递的参数。
    6. 最后把所有函数a的形参和内部的函数b的引用也添加到a的活动对象上。在这一步中,完成了函数b的的定义,因此如同第3步,函数b的作用域链被设置为b所被定义的环境,即a的作用域。

    到此,整个函数a从定义到执行的步骤就完成了。此时a返回函数b的引用给全局变量c,又函数b的作用域链包含了对函数a的活动对象的引用,也就是说b可以访问到a中定义的所有变量和函数。函数b被c引用,函数b又依赖函数a,因此函数a在返回后不会被GC回收。

      在Javascript中,如果一个对象不再被引用,那么这个对象就会被GC回收。如果两个对象互相引用,而不再被第3者所引用,那么这两个互相引用的对象也会被回收。因为函数a被b引用,b又被a外的c引用,而c又一直存在,这就是为什么函数a执行后不会被回收的原因。

      当函数b执行的时候亦会像以上步骤一样。因此,执行时b的作用域链包含了3个对象:b的活动对象、a的活动对象和window对象,如下图所示:

      

      如图所示,当在函数b中访问一个变量的时候,搜索顺序是:

    1. 先搜索自身的活动对象,如果存在则返回,如果不存在将继续搜索函数a的活动对象,依次查找,直到找到为止。
    2. 如果函数b存在prototype原型对象,则在查找完自身的活动对象后先查找自身的原型对象,再继续查找。这就是Javascript中的变量查找机制。
    3. 如果整个作用域链上都无法找到,则返回undefined。

      让我们说的更透彻一些。所谓“闭包”,就是在构造函数体内定义另外的函数作为目标对象的方法函数,而这个对象的方法函数反过来引用外层函数体中的临时变量。这使得只要目标 对象在生存期内始终能保持其方法,就能间接保持原构造函数体当时用到的临时变量值。尽管最开始的构造函数调用已经结束,临时变量的名称也都消失了,但在目标对象的方法内却始终能引用到该变量的值,而且该值只能通这种方法来访问。即使再次调用相同的构造函数,但只会生成新对象和方法,新的临时变量只是对应新的值,和上次那次调用的是各自独立的。

      下面利用一个实际例子进行分析。

    <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd"> 
    <html xmlns="http://www.w3.org/1999/xhtml"> 
    <head> 
    <title></title> 
    <script type="text/javascript"><!-- 
    window.onload = function() { 
    for (var i = 1; i < 4; i++) { 
    var id = document.getElementById("a" + i); 
    id.onclick = function(){ alert(i); }
    } 
    } 
    // --></script> 
    </head> 
    <body> 
    <ul> 
    <li id="a1">aa</li> 
    <li id="a2">aa</li> 
    <li id="a3">aa</li> 
    </ul> 
    </body> 
    </html> 

      这段代码的实际效果是无论点击哪一个标签都是现实“4”。因为在onload函数后执行完后i的值就是4了,中间变化的过程值没有被内存存储。闭包允许内层函数引用父函数中的变量,但是该变量是最终值

    <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd"> 
    <html xmlns="http://www.w3.org/1999/xhtml"> 
    <head> 
    <title></title> 
    <script type="text/javascript"><!-- 
    window.onload = function() { 
    for (var i = 1; i < 4; i++) { 
    var id = document.getElementById("a" + i); 
    id.onclick = (function(i) { 
    return function() { 
    alert(i); 
    } 
    })(i); 
    } 
    } 
    // --></script> 
    </head> 
    <body> 
    <ul> 
    <li id="a1">aa</li> 
    <li id="a2">aa</li> 
    <li id="a3">aa</li> 
    </ul> 
    </body> 
    </html> 

      而这段代码的实际效果为点击标签其会显示对应的数字,因为在onload执行的过程中利用了闭包,将每次调用函数产生的结果都保存在了内存(全局变量c)中,三个不同的i值被单独存储了。

      闭包的作用?

      在动态执行环境中,数据实时地发生变化,为了保持这些非持久型变量的值,我们用闭包这种载体来存储这些动态数据。这就是闭包的作用。也就说遇到需要存储动态变化的数据或将被回收的数据时,我们可以通过外面再包裹一层函数形成闭包来解决。闭包并不能存储动态值,需要在全局作用域中调用才能实现存储功能。

      

  • 相关阅读:
    什么是响应式设计?响应式设计的基本原理是什么?响应式设计的优缺点?
    你了解的浏览器兼容问题有哪些?
    简述网页中常见图片格式及特点?
    标签应该如何合理嵌套?
    你能想出几种方法让元素在页面中消失?
    为什么要初始化 CSS 样式?哪些样式需要初始化?
    文本溢出显示省略号如何实现?
    vue 几个典型的坑
    vue指令 v-if与 v-show的区别
    vuex的demo
  • 原文地址:https://www.cnblogs.com/lflj/p/6556986.html
Copyright © 2020-2023  润新知