zoukankan      html  css  js  c++  java
  • 浅谈JavaScript中闭包

    引言

      闭包可以说是JavaScript中最有特色的一个地方,很好的理解闭包是更深层次的学习JavaScript的基础。这篇文章我们就来简单的谈下JavaScript下的闭包。

    闭包是什么?

      闭包是什么?通俗的解释是:有权访问另一个函数作用域中变量的函数。创建闭包的常见方式,就是在一个函数内部创建另一个函数(作为其子函数)。下面我们还是以前面的一个例子来简单介绍下:

     1 //通过属性名称来对数组元素进行排序
     2     function createComparisonFunction(propertyName) {
     3         return function (obj1, obj2) {
     4             var val1 = obj1[propertyName];
     5             var val2 = obj2[propertyName];
     6             if (val1 < val2) {
     7                 return -1;
     8             }
     9             else if (val1 > val2) {
    10                 return 1;
    11             }
    12             else {
    13                 return 0;
    14             }
    15         }
    16     }

      我们看的在这个函数中我们定义了一个匿名函数,并且将匿名函数作为值返回。注意代码地4、5行,这两行代码访问了外部函数中的变量propertyName。即使这个函数返回了,或者在其他地方被调用了,我们通过这个匿名函数仍然可以访问这个变量。这是为什么呢?想想前面在对象内部搜索属性的机制。很明显,匿名函数的作用域链包含了createComparisonFunction的作用域链。这又是为什么呢?还得从函数被第一次调用发生的一些细节上进行讨论。

     函数第一次调用到底发生了什么

      之前介绍作用域链的博客中有关于这方面内容的介绍。相信大家肯定还记忆犹新。当一个函数第一次被调用的时候,会创建一个执行环境和作用域链,并把作用域链赋值给一个特殊的内部属性[Scope]。然后使用this、arguments和其他命名参数的值来初始化函数的活动对象。外部函数的活动对象位于第二位,外部函数的外部函数的活动对象在第三位,直到作为作用域链终点的全局执行换环境。

      下面还是通过一个简单的例子来重温下这方面的内容:

     1 function compare(value1, value2) {
     2         if (value1 < value2) {
     3             return -1;
     4         } else if (value1 == value2) {
     5             return 0;
     6         }
     7         else {
     8             return 1;
     9         }
    10     }
    11 
    12     var result = compare(5, 10);

      那么,按照我们之前的描述。在执行第12行代码的时候,作用域链相关的分析应该是这样的。看图:

      后台的每一个执行环境都有一个表示变量的对象--变量对象。全局环境的变量对象始终存在,像compare函数这样的局部环境的变量对象,只是在运行时存在。在创建compare()函数的时候,会创建一个预先包含全局变量对象的作用域链。这个作用域链被包含在内部的[Scope]属性中。当调用compare()函数的时候,会为函数创建一个执行环境,然后通过复制函数的[Scope]属性中的对象构建起执行环境的作用域链。

      无论什么时候在函数中访问一个变量时,就会从作用域链中搜索具有相应名字的变量。一般来讲,当函数执行完毕以后,局部变量对象就会被销毁,内存中只保存全局作用域。但是闭包让情况变的有点不同。

       下面我们来看下我们最开始的那个例子:

    1 var compare = createComparisonFunction("name");
    2 var result = compare({ name: "Nicolas" }, { name: "Grey" });

      在另一个函数内部会将包含函数(外部函数)的活动对象添加到它的作用域链中。因此在createComparisonFunction函数内部定义的匿名函数的作用域链中会将createComparisonFunction函数的变量对象包含在自己的作用域链中。下面这张图很好的展示了这一点:

      匿名函数的作用域链中引用了外部函数的变量对象(活动对象)。但是:createComparisonFunction函数执行完以后,其活动对象也不会被销毁。因为匿名函数的作用域链中还引用着createComparisonFunction的活动对象。我们也可以这样认为,createComparisonFunction函数执行完以后,其作用域链被销毁,但是其活动对象仍然在内存中。所以,过度的使用闭包可能会导致内存占用过高。

      闭包与变量

      作用域链的这种配置机制导致了一个副作用,即闭包只能取得包含函数中任意变量的最后一个值。因为闭包通过作用域链引用的是整个变量对象。外部函数的变量存储在其变量对象中。下面的例子可以展示这个问题:

     1 /**
     2      * 闭包与变量的关系示例
     3      **/
     4     function createFunctions() {
     5         var result = [];
     6         for (var i = 0; i < 10; i++) {
     7             result[i] = function () {
     8                 return i;
     9             }
    10         }
    11         return result;
    12     }
    13 
    14     var funs = createFunctions();
    15     for (var i = 0; i < 10; i++) {
    16         alert(funs[i]());           //输出10次10
    17     }

      我们看到每一个函数都输出10。并不是我们想象中的1-10之间的数值。因为每一个result数组引用的匿名函数内部都包含了createFunctions函数的活动对象。循环每一次的调用,修改的都是createFunctions变量对象中的i值。最后我们调用的时候看到的只是最后的一个i的值。那么我们怎么修改,才能按预想的输出1-10呢。问题的关键在于:我们如果能每一次循环的时候把i的值预存起来不就可以了吗?看看下面的这个改进方案:

     1 function createFunctions() {
     2         var result = [];
     3         for (var i = 0; i < 10; i++) {
     4             result[i] = (function (argument) {
     5                 return function () {
     6                     return argument;
     7                 }
     8             })(i);
     9         }
    10         return result;
    11     }
    12 
    13     var funs = createFunctions();
    14     for (var i = 0; i < 10; i++) {
    15         alert(funs[i]());           //输出1-9
    16     }

      我们通过改进后,终于如愿的输出了1-9。看看到底发生了什么?在代码的第4-8行,我们看到我们创建了一个匿名函数,并且将i的值作为参数传递给它,然后立即执行这个匿名函数。这个匿名函数内部返回了另一个匿名函数,result数组中保存的匿名函数的作用域链里面就会有4个活动对象,分别是本身的活动对象、外部匿名函数(已执行)的活动对象(包含传递的i的值,即argument)、createFunctions的活动对象、全局活动对象。下面我们在执行返回的匿名函数时,通过作用域链来搜索到argument变量。每一个argument变量都是当时执行时传递的i的值。

      关于this对象

      在闭包中使用this值也会导致一些问题。this对象是在运行时根据函数的执行环境绑定的。在全局执行环境中,this等于window,而当函数作为某一个对象的方法调用时,this等于那个对象。匿名函数的执行环境具有全局性,this的值通常等于window。但有时候,可能由于编写闭包的方式不同,这一点可能不会那么明显。比如下面的例子:

     1 var name = "The Window";
     2     var object = {
     3         name: "The Object",
     4         getNameFun: function () {
     5             return function () {
     6                 return this.name;
     7             }
     8         }
     9     }
    10 
    11     alert(object.getNameFun()());   //输出The Window

      按照之前在作用域链中搜索变量的机制。输出应该是The Object才对。但是为什么是The Window呢?前面应该提到过,每个函数在被调用时,其活动对象都会自动获取两个变量this和arguments。内部函数在搜索这两个变量的时候,只会搜索到其活动对象为止,因此,无法访问外部函数的这两个变量。不过通过简单的修改我们可以实现弹出The Object的效果。请看下面的例子:

     1 var name = "The Window";
     2     var object = {
     3         name: "The Object",
     4         getNameFun: function () {
     5             var that = this;
     6             return function () {
     7                 return that.name;
     8             }
     9         }
    10     }
    11 
    12     alert(object.getNameFun()());   //输出The Object

      我们在返回匿名函数之前,将this保存在that变量中,作为闭包,最深层次的匿名函数在调用时,其作用域链中会包含getNameFun这个函数的活动对象。因此这时that还是引用object对象。我们能正常的弹出The Object。讲到这里,相信大家对闭包都有一个详细的了解了把。最后推荐大家看个网页,里面有很多经典的闭包的事例哦。http://www.oschina.net/question/28_41112

  • 相关阅读:
    策略思维模式
    初学者之心
    《计算机思维》笔记
    通才论
    Shell 脚本中 set ex 命令的作用
    Python—requests模块详解
    SharePoint CSOM 迁移列表项权限
    实践剖析.NET Core如何支持Cookie和JWT混合认证、授权
    面试突击13:方法优先调用可选参数还是固定参数?
    如何构建一个疾病的动物模型? | animal model of disease
  • 原文地址:https://www.cnblogs.com/dreamGong/p/4931570.html
Copyright © 2011-2022 走看看