`

JavaScript 作用域链解析

 
阅读更多

 

最近看了下JavaScript方面的几本书,把里面的一些核心概念按照自己的理解做个总结。

 

JavaScript 中有 Scope( 作用域 ) Scope chain( 作用域链 ) Execute context( 执行上下文 ) Active Object ( 活动对象 ),Dynamic Scope( 动态作用域 ) Closure( 闭包 ) 这些概念,要理解这些概念,我们从静态和动态两个方面去分析一下。

首先我们写一个简单的 function 来做一个例子:

function add(num1, num2){

var sum = num1 + num2;

return sum;

}

我们定义了一个具有两个形参的 add 函数。

静态方面:

当创建 add 函数的时候, Javascript 引擎会创建 add 函数的 Scope chain, 这个作用域链指向了 Global Context( 全局上下文 ) 。如果用图形形象化的表述如下图所示:


 

从上图可以看出,当 add 函数创建的时候,作用域链就已经创建了,因此可以得出一个结论,函数的作用域链是创建函数的时候就已经创建了,而不是动态运行期。下面就来看看动态运行期的时候会发生什么事情。

 

动态方面:

当执行 add 函数的时候, JavaScript 会创建一个 Execute context (执行上下文),执行上下文中就包含了 add 函数运行期所需要的所有信息。 Execute context 也有自己的 Scope chain, 当函数运行的时候, JavaScript 引擎会首先从用 add 函数的作用域链来初始化执行上下文的作用域链,然后 JavaScript 引擎又会创建一个 Active Object, 这个对象里面包含了函数运行期的所有局部变量,参数以及 this 等变量。

如果形象的描述 add 函数动态运行期会发生什么,可以用如下图来描述:



 从上图可以看出,执行上下文是一个动态的概念,它是当函数运行的时候创建的,同时 Active Object 对象也是一个动态的概念,它是被执行上下文的作用域链引用的。因此可以得出一个结论:执行上下文和活动对象都是动态概念,并且执行上下文的作用域链是由函数作用域链初始化的。

上面说了函数作用域和执行上下文作用域,下面接着说一下动态作用域的问题,当在 JavaScript 通过 with 语句, try-catch catch 子句,以及 eval 方法的时候, JavaScript 引擎就会动态的改变执行上下文的作用域。下面还是通过一个例子来看看:

function initUI(){

with (document){ //avoid!

var bd = body,

links = getElementsByTagName("a"),

i= 0,

len = links.length;

while(i < len){

update(links[i++]);

}

getElementById("go-btn").onclick = function(){

start();

};

bd.className = "active";

}

当执行上面的 initUI 函数的时候, JavaScript 会动态的创建一个 with 语句对应的作用域放到执行上下文作用域链的最前端,通过下图可以形象的描述上述过程,下图红色标注的区域就显示了 with 语句产生的作用域。



 最后,我们来看看 JavaScript 最神秘的 Closure (闭包),闭包在 JavaScript 其实就是一个函数,闭包是在函数运行期被创建的,下面还是以一个实例来看看:

function assignEvents(){

var id = "xdi9592";

document.getElementById("save-btn").onclick = function(event){

saveDocument(id);

};

}

当上面的 assignEvents 函数被执行的时候,会创建一个闭包,而这个闭包会引用 assignEvents 作用域中的 id 变量,如果按照传统的编程语言的方式, id 是存储在堆栈上的一个变量,当函数执行完了以后 id 就消失,那么怎么可能再次引用呢?显然这里 JavaScript 采用了另外的方式。下面就来看看 JavaScript 是如何来实现闭包的。当执行 assignEvents 函数的时候, JavaScript 引擎会创建assignEvents函数执行上下文的作用域链,这个作用域链包含了 assignEvents 执行时的活动对象,而同时 JavaScript 引擎也会创建一个闭包,而闭包的作用域链也会引用 assignEvent 执行时候的活动对象,这样当 assignEvents 执行完的时候,虽然它本身执行上下文的作用域链不再引用活动对象了,但是闭包还是引用着 assignEvents 运行期对应的活动对象,这就解释了 JavaScipt 内部的闭包机制。可以用下图形象的表述上面 assignEvents 函数运行期的情形:



    从上面可以看出,当 assignEvents 函数执行完毕以后, document.getElementById("save-btn").onclick 引用了闭包,这样当用户点击 save-btn 的时候,就会触发闭包的执行,那么下面就来看看闭包执行时的情形。前面也说了 JavaScript 中闭包其实就是函数,因此闭包执行和函数执行时的情形是一致的,通过下图来形象的描述上述 onclick 事件所关联的闭包。

 

 



 从上图可以看出 JavaScript 引擎首先创建了闭包的执行上下文,然后用闭包作用域链来初始化闭包的执行上下文作用域链,最后再将闭包执行时对应的活动对象放入到作用域的最前端,这也进一步验证了闭包就是函数的论断。

 

参考资料:

1.High Performance JavaScript. http://book.douban.com/subject/5362856/

2.JavaScript高级程序设计. http://book.douban.com/subject/4886879/

 

 

 

 

 

  • 大小: 21 KB
  • 大小: 35.2 KB
  • 大小: 50.4 KB
  • 大小: 43.4 KB
  • 大小: 43.4 KB
1
0
分享到:
评论
3 楼 ynyee 2012-05-27  
那闭包的生成以后怎么销毁?
2 楼 狂放不羁 2011-12-15  
oojdon 写道
我现在的理解是代码被执行,就形成了闭包,代码没执行那就还是代码

你的理解是对的。哥们。
1 楼 oojdon 2011-12-15  
我现在的理解是代码被执行,就形成了闭包,代码没执行那就还是代码

相关推荐

    深入理解JavaScript作用域和作用域链

    深入理解JavaScript作用域和作用域链对于编写高效、无错的代码至关重要。正确管理作用域可以避免全局变量冲突,提高代码的复用性和模块化,同时也有助于提升性能,因为局部变量的访问速度通常比全局变量快。在实际...

    javascript作用域链(Scope Chain)初探.docx

    ### JavaScript作用域链(Scope Chain)初探 #### 一、引言 JavaScript的作用域链是一个重要的概念,尤其是在深入理解JavaScript执行机制时不可或缺的一部分。本文将通过对几个具体例子的分析来探讨JavaScript作用域...

    javascript作用域链(Scope Chain)用法实例解析

    JavaScript 在解析代码时,会将 `var` 声明的变量和函数声明提升到它们所在作用域的顶部,但不会提升赋值操作。这解释了为什么以下代码会输出 "undefined": ```javascript var arg = 1; function funcTest() { ...

    JavaScript — 原型链与作用域链1

    JavaScript是一种广泛用于网页和网络应用的脚本语言,它的核心特性包括原型链和作用域链。这两个概念是理解JavaScript中对象继承和变量访问的关键。 **作用域链** 1. **作用域生成**:每次JavaScript代码执行时,...

    JavaScript作用域与作用域链深入解析

    作用域是JavaScript最重要的概念之一,想要学好JavaScript就需要理解JavaScript作用域和作用域链的工作原理。今天这篇文章对JavaScript作用域和作用域链作简单的介绍,希望能帮助大家更好的学习JavaScript。 ...

    Javascript作用域和作用域链原理解析

    理解JavaScript的作用域和作用域链对于编写健壮、可维护的代码至关重要。它可以帮助开发者更好地管理变量,减少错误,并提高代码的可读性和可预测性。深入掌握这些概念,将使你成为一名更出色的JavaScript开发者。

    图解javascript作用域链

    JavaScript的作用域链是理解JavaScript执行环境的关键概念,它决定了变量和函数的可访问性。在JavaScript中,每个函数都有自己的作用域,而这些作用域按照特定的顺序组织起来,形成了作用域链。这个链帮助解析器在...

    深入了解JavaScript,从作用域链开始(1).pdf

    总的来说,深入理解JavaScript的作用域和作用域链对于编写高效、无bug的代码至关重要。通过掌握这些基本概念,开发者可以更好地管理变量和函数的生命周期,避免全局变量的滥用,并有效地组织代码逻辑。

    JavaScript: 函数与作用域深入解析及应用场景

    接着讨论了 JavaScript 中的全局作用域、局部作用域、块级作用域和函数作用域,特别是闭包的概念。随后,文章探讨了函数的高级用法,如递归函数、高阶函数和立即执行函数表达式(IIFE)。最后,通过实际应用示例,如...

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

    以上就是对"深入理解JavaScript作用域"的详细解析,涵盖了一系列关键知识点。理解并掌握这些概念对于编写高效、可靠的JavaScript代码至关重要。在实际编程中,合理利用作用域可以避免变量冲突,提高代码质量,确保...

    浅析JavaScript作用域链、执行上下文与闭包

    当我们查找一个变量时,JavaScript会从当前作用域开始,沿着作用域链向上搜索,直到找到变量或者到达链的末端。 **执行上下文**是理解JavaScript执行流程的关键。每当代码执行时,都会创建一个新的执行上下文。全局...

    JavaScript程序设计-变量作用域.pdf

    作用域链是JavaScript中实现作用域机制的一种方式,它是由一系列作用域构成的链状结构,每个函数都有自己的作用域链,用于在查找变量时提供路径。当尝试访问一个变量时,JavaScript会首先在当前作用域查找,如果没有...

    javascript 作用于作用域链的详解

    JavaScript作用域和作用域链是JS编程中非常核心的概念,理解这些概念对于编写可预测和高效的代码至关重要。首先,让我们来探讨JavaScript中的作用域。 一、JavaScript作用域 在JavaScript中,作用域决定了变量和...

    深入理解变量作用域

    - 当JavaScript引擎需要查找变量时,它会从当前作用域开始,沿着作用域链向上查找,直到找到相应的变量为止。 - 如果在当前作用域找不到变量,则继续在上一层作用域中查找,依此类推,直到到达全局作用域。 - 这...

Global site tag (gtag.js) - Google Analytics