javascript之闭包深刻理解(二)

在上一节中,详细理解了做用域链和垃圾回收机制,彷佛这两点跟闭包关系不大,可是仔细想想就会发现,其实否则。这一节将经过上一部分的说明详细理解闭包。请看代码:前端

function createComparisonFunction(propertyName) {
    return function(object1, object2){
        var value1 = object1[propertyName]; var value2 = object2[propertyName]; if (value1 < value2){
            return -1;
        } else if (value1 > value2){
            return 1;
        } else {
            return 0;
        }
    };
}            

  在这个例子中,突出的那两行代码是内部函数(一个匿名函数)中的代码,这两行代码访问了外部函数中的变量propertyName。即便这个内部函数被返回了,并且是在其余地方被调用了,但它仍然能够访问变量propertyName。之因此还可以访问这个变量,是由于内部函数的做用域链中包含createComparisonFunction()的做用域。数组

  当某个函数被调用时,会建立一个执行环境(execution context)及相应的做用域链。而后,使用arguments 和其余命名参数的值来初始化函数的活动对象(activation object)。但在做用域链中,外部函数的活动对象始终处于第二位,外部函数的外部函数的活动对象处于第三位,……直至做为做用域链终点的全局执行环境。闭包

  在函数执行过程当中,为读取和写入变量的值,就须要在做用域链中查找变量。函数

function compare(value1, value2){
    if (value1 < value2){
        return -1;
    } else if (value1 > value2){
        return 1;
    } else {
        return 0;
    }
}
var result = compare(5, 10);

  以上代码先定义了compare()函数,而后又在全局做用域中调用了它。当调用compare()时,会建立一个包含arguments、value1 和value2 的活动对象。全局执行环境的变量对象(包含result和compare)在compare()执行环境的做用域链中则处于第二位。图7-1 展现了包含上述关系的compare()函数执行时的做用域链。优化

  后台的每一个执行环境都有一个表示变量的对象——变量对象。全局环境的变量对象始终存在,而像compare()函数这样的局部环境的变量对象,则只在函数执行的过程当中存在。在建立compare()函数时,会建立一个预先包含全局变量对象的做用域链,这个做用域链被保存在内部的[[Scope]]属性中。当调用compare()函数时,会为函数建立一个执行环境,而后经过复制函数的[[Scope]]属性中的对象构建起执行环境的做用域链。此后,又有一个活动对象(在此做为变量对象使用)被建立并被推入执行环境做用域链的前端。对于这个例子中compare()函数的执行环境而言,其做用域链中包含两个变量对象:本地活动对象和全局变量对象。显然,做用域链本质上是一个指向变量对象的指针列表,它只引用但不实际包含变量对象。spa

  不管何时在函数中访问一个变量时,就会从做用域链中搜索具备相应名字的变量。通常来说,当函数执行完毕后,局部活动对象就会被销毁,内存中仅保存全局做用域(全局执行环境的变量对象)。可是,闭包的状况又有所不一样。指针

  在另外一个函数内部定义的函数会将包含函数(即外部函数)的活动对象添加到它的做用域链中。所以,在createComparisonFunction()函数内部定义的匿名函数的做用域链中,实际上将会包含外部函数createComparisonFunction()的活动对象。code

  在匿名函数从createComparisonFunction()中被返回后,它的做用域链被初始化为包含createComparisonFunction()函数的活动对象和全局变量对象。这样,匿名函数就能够访问在createComparisonFunction()中定义的全部变量。更为重要的是,createComparisonFunction()函数在执行完毕后,其活动对象也不会被销毁,由于匿名函数的做用域链仍然在引用这个活动对象。换句话说,当createComparisonFunction()函数返回后,其执行环境的做用域链会被销毁,但它的活动对象仍然会留在内存中;直到匿名函数被销毁后,createComparisonFunction()的活动对象才会被销毁。对象

  因为闭包会携带包含它的函数的做用域,所以会比其余函数占用更多的内存。过分使用闭包可能会致使内存占用过多,咱们建议读者只在绝对必要时再考虑使用闭包。虽然像V8 等优化后的JavaScript 引擎会尝试回收被闭包占用的内存,但请你们仍是要慎重使用闭包。blog

  继续

function createFunctions(){
    var result = new Array();
        for (var i=0; i < 10; i++){
        result[i] = function(){
            return i;
        };
    }
    return result;
}

  这个函数会返回一个函数数组。表面上看,彷佛每一个函数都应该返本身的索引值,即位置0 的函数返回0,位置1 的函数返回1,以此类推。但实际上,每一个函数都返回10。由于每一个函数的做用域链中都保存着createFunctions() 函数的活动对象, 因此它们引用的都是同一个变量i 。当createFunctions()函数返回后,变量i 的值是10,此时每一个函数都引用着保存变量i 的同一个变量对象,因此在每一个函数内部i 的值都是10

  

function createFunctions(){
    var result = new Array();
    for (var i=0; i < 10; i++){
        result[i] = function(num){
            return function(){
                return num;
            };
        }(i);
    }
    return result;
}    

  在重写了前面的createFunctions()函数后,每一个函数就会返回各自不一样的索引值了。在这个版本中,咱们没有直接把闭包赋值给数组,而是定义了一个匿名函数,并将当即执行该匿名函数的结果赋给数组。这里的匿名函数有一个参数num,也就是最终的函数要返回的值。在调用每一个匿名函数时,咱们传入了变量i。因为函数参数是按值传递的因此就会将变量i 的当前值复制给参数num。而在这个匿名函数内部,又建立并返回了一个访问num 的闭包。这样一来,result 数组中的每一个函数都有本身num 变量的一个副本,所以就能够返回各自不一样的数值了。

相关文章
相关标签/搜索