`

Javascript 的词法作用域、调用对象和闭包

阅读更多
引用

作用域 Scope

Javascript 中的函数属于词法作用域,也就是说函数在它被定义时的作用域中运行而不是在被执行时的作用域内运行。这是犀牛书上的说法。但“定义时”和“执行(被调用)时”这两个东西有些人搞不清楚。简单来说,一个函数A在“定义时”就是 function A(){} 这个语句执行的时候就是定义这个函数的时候,而A被调用的时候是 A() 这个语句执行的时候。这两个概念一定要分清楚。

那词法作用域(以下称之为“作用域”,除非特别指明)到底是什么呢?它是个抽象的概念,说白了它就是一个“范围”,scope 在英文里就是范围的意思。一个函数的作用域是它被定义时它所处的“范围”,也就是它外层的“范围”,这个“范围”包含了外层的变量属性,这个“范围”被设置成这个函数的一个内部状态。一个全局函数被定义的时候,全局(这个函数的外层)的“范围”就被设置成这个全局函数的一个内部状态。一个嵌套函数被定义的时候,被嵌套函数(外层函数)的“范围”就被设置成这个嵌套函数的一个内部状态。这个“内部状态”实际上可以理解成作用域链,见下文。

当一个函数被调用时,在这个函数里,它能访问到它的这个内部状态,也就可以访问整个作用域链上的所有变量,当然也就包括了外部变量。(实际上是从“调用对象链”里访问到的。好像有问题吧,请接着阅读。)

照以上说法,一个函数的作用域是它被定义的时候所处的“范围”,那么 Javascript 里的函数作用域是在函数被定义的时候就确定了,所以它是静态的作用域,词法作用域又称为静态作用域。
调用对象 Call Object

一个函数的调用对象是动态的,它是在这个函数被调用时才被实例化的。我们已经知道,当一个函数被定义的时候,已经确定了它的作用域链。当 Javascript 解释器调用一个函数的时候,它会添加一个新的对象(调用对象)到这个作用域链的前面。这个调用对象的一个属性被初始化成一个名叫 arguments 的属性,它引用了这个函数的 Arguments 对象,Arguments 对象是函数的实际参数。所有用 var 语句声明的本地变量也被定义在这个调用对象里。这个时候,调用对象处在作用域链的头部,本地变量、函数形式参数和 Arguments 对象全部都在这个函数的范围里了。当然,这个时候本地变量、函数形式参数和 Arguments 对象就覆盖了作用域链里同名的属性。
作用域、作用域链和调用对象之间的关系

我的理解是,作用域是是抽象的,而调用对象是实例化的。

在函数被定义的时候,实际上也是它外层函数执行的时候,它确定的作用域链实际上是它外层函数的调用对象链;当函数被调用时,它的作用域链是根据定义的时候确定的作用域链(它外层函数的调用对象链)加上一个实例化的调用对象。所以函数的作用域链实际上是调用对象链。在一个函数被调用的时候,它的作用域链(或者称调用对象链)实际上是它在被定义的时候确定的作用域链的一个超集。

它们之间的关系可以表示成:作用域⊃作用域链⊇调用对象。

太绕口了,举例说明吧:

function f(x) {
    var g = function () { return x; }
    return g;
}
var g1 = f(1);
alert(g1());  //输出 1


假设我们把全局看成类似以下这样的一个大匿名函数:
(function() {
    //这里是全局范围
})();


那么例子就可以看成是:
(function() {
    function f(x) {
        var g = function () { return x; }
        return g;
    }
    var g1 = f(1);
    alert(g1());  //输出 1
})();


   0. 全局的大匿名函数被定义的时候,它没有外层,所以它的作用域链是空的。
   1. 全局的大匿名函数直接被执行,全局的作用域链里只有一个 '全局调用对象'。
   2. 函数 f 被定义,此时函数 f 的作用域链是它外层的作用域链,即 '全局调用对象'。
   3. 函数 f(1) 被执行,它的作用域链是新的 f(1) 调用对象加上函数 f 被定义的时候的作用域链,即 'f(1) 调用对象->全局调用对象'。
   4. 函数 g (它要被返回给 g1,就命名为 g1吧)在 f(1) 中被定义,它的作用域链是它外层的函数 f(1) 的作用域链,即 'f(1) 调用对象->全局调用对象'。
   5. 函数 f(1) 返回函数 g 的定义给 g1。
   6. 函数 g1 被执行,它的作用域链是新的 g(1) 调用对象加上外层 f(1) 的作用域链,即 'g1 调用对象->f(1)调用对象->全局调用对象'。

这样看就很清楚了吧。
闭包 Closuer

闭包的一个简单的说法是,当嵌套函数在被嵌套函数之外调用的时候,就形成了闭包。

之前的那个例子其实就是一个闭包。g1 是在 f(1) 内部定义的,却在 f(1) 返回后才被执行。可以看出,闭包的一个效果就是被嵌套函数 f 返回后,它内部的资源不会被释放。在外部调用 g 函数时,g 可以访问 f 的内部变量。根据这个特性,可以写出很多优雅的代码。

例如要在一个页面上作一个统一的计数器,如果用闭包的写法,可以这么写:

var counter  = (function() {
    var i = 0;
    var fns = {"get": function() {return i;},
               "inc": function() {return ++i;}};
    return fns;
})();
//do something
counter.inc();
//do something else
counter.inc();
var c_value = counter.get();  //now c_value is 2


这样,在内存中就维持了一个变量 i,整个程序中的其它地方都无法直接操作 i 的值,只能通过 counter 的两个操作。

在 setTimeout(fn, delay) 的时候,我们不能给 fn 这个函数句柄传参数,但可以通过闭包的方法把需要的参数绑定到 fn 内部。

for(var i=0,delay=1000; i< 5; i++, delay +=1000) {
    setTimeout(function() {
        console.log('i:' + i + " delay:" + delay);
    }, delay);
}


这样,打印出来的值都是
delay:6000
i:5 delay:6000
i:5 delay:6000
i:5 delay:6000
i:5 delay:6000


改用闭包的方式可以很容易绑定要传进去的参数:
for(var i=0, delay=1000; i < 5; i++, delay += 1000) {
    (function(a, _delay) { 
        setTimeout(function() { 
            console.log('i:'+a+" delay:"+_delay);
        }, _delay);
    })(i, delay);
}


输出:
i:0 delay:1000
i:1 delay:2000
i:2 delay:3000
i:3 delay:4000
i:4 delay:5000


闭包还有一个很常用的地方,就是在绑定事件的回调函数的时候。也是同样的道理,绑定的函数句柄不能做参数,但可以通过闭包的形式把参数绑定进去。
总结

   1. 函数的词法作用域和作用域链是不同的东西,词法作用域是抽象概念,作用域链是实例化的调用对象链。
   2. 函数在被定义的时候,同时也是它外层的函数在被执行的时候。
   3. 函数在被定义的时候它的词法作用域就已经确定了,但它仍然是抽象的概念,没有也不能被实例化。
   4. 函数在被定义的时候还确定了一个东西,就是它外层函数的作用域链,这个是实例化的东西。
   5. 函数在被多次调用的时候,它的作用域链都是不同的。
   6. 闭包很强大。犀牛书说得对,理解了这些东西,你就可以自称是高级 Javascript 程序员了。因为利用好这些概念,可以玩转 Javascript 的很多设计模式。

-EOF-
分享到:
评论

相关推荐

    JavaScript词法作用域与调用对象深入理解

    JavaScript中的词法作用域和调用对象是理解其运行机制的关键概念。词法作用域指的是函数在其被定义时所在的作用域内运行,而非执行时。这意味着函数能够访问在其定义时可及的所有变量和函数,即使在它被调用时这些...

    javascript 词法作用域和闭包分析说明

    通过以上知识点,我们可以对JavaScript中词法作用域和闭包的行为有更清晰的认识。理解这些概念对于编写可维护和高效的JavaScript代码是非常重要的,尤其是在处理异步回调函数或需要保护数据不被外部直接访问的情况下...

    一篇文章搞懂:词法作用域、动态作用域、回调函数及闭包

    在编程领域,理解和掌握词法作用域、动态作用域、回调函数以及闭包是非常重要的概念,它们直接影响着代码的执行逻辑和内存管理。下面我们将详细探讨这些主题。 首先,词法作用域(Lexical Scoping)是一种变量作用...

    网易JS面试题与Javascript词法作用域说明

    JavaScript中的词法作用域是编程中的一个重要概念,它决定了变量的可访问性和作用范围。根据题目描述,我们来深入理解这一知识点。 首先,JavaScript的词法作用域在函数定义时就已经确定,而不是在函数执行时。这...

    深入理解javascript作用域第二篇之词法作用域和动态作用域

    词法作用域使代码更易于预测和调试,因为它遵循固定的查找规则,而动态作用域可能导致意外的行为,特别是在函数嵌套和闭包的情况下。因此,在JavaScript开发中,了解并掌握词法作用域是非常重要的。

    夯实基础中篇-图解作用域链和闭包.doc

    最后,**闭包**是函数能够记住并访问其词法作用域内变量的能力,即使该函数已经返回并且其外部词法作用域不再存在。在以下示例中: ```javascript var globalVariable = 1 const scriptVariable = 2 function test...

    JavaScript作用域、闭包、对象与原型链概念及用法实例总结

    - **装饰的定义**:闭包是一种特殊的现象,它使得函数能够记住其词法作用域,即使该函数已经执行完毕。闭包允许函数访问并操作其外部作用域内的变量,即使这些变量在函数外部不再存在。闭包的主要用途包括: - **...

    005课-继承作用域闭包.rar

    此外,JavaScript还引入了块级作用域的概念(通过let和const关键字),以及函数作用域和词法作用域的概念。词法作用域是指函数的执行上下文由其定义时的位置决定,而不是调用时的位置。 **闭包** 闭包是JavaScript...

    基于javascript 闭包基础分享

    在JavaScript编程中,闭包是一个核心概念,它允许函数记住并访问所在词法作用域,即使当函数在其词法作用域之外执行时。闭包的特性使得它在JavaScript中既神秘又强大。 首先,我们从闭包的定义谈起。在JavaScript中...

    深入了解JavaScript,优化作用域链(2).pdf

    最后,作用域链和作用域优化的知识点还可以涉及到词法作用域、动态作用域、变量提升、作用域提升、立即调用函数表达式(IIFE)等高级概念。学习和掌握这些概念对于编写高性能的JavaScript代码至关重要。

    深入理解JavaScript作用域共12页.pdf.zip

    此外,JavaScript还引入了闭包,这是一种特殊的作用域现象,允许函数访问并操作其词法作用域内的变量,即使该函数在其外部被调用。闭包在实现数据封装和模块化编程时非常有用。 作用域规则还涉及到变量声明。在ES5...

    深入理解javascript原型和闭包1

    闭包是JavaScript中一种强大的特性,它允许函数访问并操作其词法作用域内的变量,即使该函数已经执行完毕。闭包形成的原因是函数可以记住其创建时的作用域,即使函数的执行上下文已经改变。 (16)——完结 至此,...

    第四章示例代码__对象的作用域

    5. **静态作用域**:静态作用域(也称为词法作用域)是指对象的可见性基于它被声明的位置,而不是调用它的位置。这意味着,如果一个对象在某个函数内部定义,但在外部被引用,编译器会查找最近的封闭作用域来解析...

    JavaScript进阶(二)词法作用域与作用域链实例分析

    在JavaScript中,词法作用域和作用域...理解词法作用域和作用域链对于解决JavaScript中的变量访问问题至关重要,尤其是在处理闭包和异步操作时。它们可以帮助开发者避免变量污染和命名冲突,提高代码的可读性和维护性。

    浅谈JavaScript作用域和闭包

    在JavaScript中,闭包是一种特殊的对象,它记录了创建它的函数所在的词法作用域,即使该函数已经执行完毕。闭包的这一特性使得它能够访问并操作函数外部的变量。这在某些情况下非常有用,例如,可以用来创建工厂函数...

Global site tag (gtag.js) - Google Analytics