JavaScript中的作用域链和闭包

JavaScript中的作用域链和闭包

2012-06-29 11:41 1878人阅读 评论(46) 收藏 举报

JavaScript中出现了一个以前没学过的概念——闭包。何为闭包?从表面理解即封闭的包,与作用域有关。所以,说闭包以前先说说作用域。

作用域(scope)

通常来说一段程序代码中使用的变量和函数并不总是可用的,限定其可用性的范围即作用域,作用域的使用提高了程序逻辑的局部性,增强程序的可靠性,减少名字冲突。

全局作用域(Global Scope)

在代码中任何地方都能访问到的对象拥有全局作用域,以下几种情形拥有全局作用域:

1、最外层函数和在最外层函数外面定义的变量拥有全局作用域,例如:

[javascript] view plaincopyprint?

  1. <span style="font-family:KaiTi_GB2312;font-size:18px;">var outSide="var outside";
  2. function outFunction(){
  3. var name="var inside";
  4. function inSideFunction(){
  5. alert(name);
  6. }
  7. inSideFunction();
  8. }
  9. alert(outSide); //正确
  10. alert(name); //错误
  11. outFunction(); //正确
  12. inSideFunction() //错误</span>

2、未定义直接赋值的变量自动声明为拥有全局作用域,例如:

[javascript] view plaincopyprint?

  1. <span style="font-family:KaiTi_GB2312;font-size:18px;">blogName="CSDN李达"</span>

3、所有window对象的属性拥有全局作用域,例如:window对象的内置属性都拥有全局作用域,例如window.name、window.location、window.top等

局部作用域(Local Scope)

[javascript] view plaincopyprint?

  1. <span style="font-family:KaiTi_GB2312;font-size:18px;">function outFunction(){
  2. var name="inside name";
  3. function inFunction(){
  4. alert(name);
  5. }
  6. inFunction();
  7. }
  8. alert(name); //错误
  9. inFunction(); //错误</span>

作用域链(scope chain)

JavaScript中,JavaScript里一切都是对象,包括函数。函数对象和其它对象一样,拥有可以通过代码访问的属性和一系列仅供JavaScript引擎访问的内部属性。其中一个内部属性是作用域,包含了函数被创建的作用域中对象的集合,称为函数的作用域链,它决定了哪些数据能被函数访问。
  当一个函数创建后,它的作用域链会被创建此函数的作用域中可访问的数据对象填充。例如函数:

[javascript] view plaincopyprint?

  1. <span style="font-family:KaiTi_GB2312;font-size:18px;">function add(num1,num2) {
  2. var sum = num1 + num2;
  3. return sum;
  4. }</span>

在函数add创建时,它的作用域链中会填入一个全局对象,该全局对象包含了所有全局变量,如下图所示(注意:图片只例举了全部变量中的一部分):

由此可见,函数的作用域链是创建函数的时候创建的。

执行上下文(Execute context )

函数add的作用域将会在执行时用到,例如:

[javascript] view plaincopyprint?

  1. <span style="font-family:KaiTi_GB2312;font-size:18px;">var total = add(5,10);</span>

当执行 add 函数的时候, JavaScript 会创建一个 Execute context (执行上下文),执行上下文中就包含了 add 函数运行期所需要的所有信息。 Execute context 也有自己的 Scope chain, 当函数运行时, JavaScript 引擎会首先从用 add 函数的作用域链来初始化执行上下文的作用域链。

活动对象(Active Object)

然后 JavaScript 引擎又会创建一个 Active Object, 这些值按照它们出现在函数中的顺序被复制到运行期上下文的作用域链中,它们共同组成了一个新的对象——“活动对象(activation object)”,这个对象里面包含了函数运行期的所有局部变量,参数以及 this 等变量,此对象会被推入作用域链的前端,当运行期上下文被销毁,活动对象也随之销毁。新的作用域链如下图所示:

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

在函数执行过程中,每遇到一个变量,都会检索从哪里获取和存储数据,该过程从作用域链头部,也就是从活动对象开始搜索,查找同名的标识符,如果找到了就使用这个标识符对应的变量,如果没有则继续搜索作用域链中的下一个对象,如果搜索完所有对象都未找到,则认为该标识符未定义,函数执行过程中,每个标识符都要经历这样的搜索过程。

闭包(closure)

先来看一个实例,javascript代码:

[javascript] view plaincopyprint?

  1. <span style="font-family:KaiTi_GB2312;font-size:18px;">    <script type="text/javascript">
  2. function newLoad(){ //新建页面加载的事件
  3. for (var i = 1; i <=3; i++) {
  4. var anchor = document.getElementById("anchor" + i); //找到每个anchor
  5. anchor.onclick = function () {//为anchor添加单击事件
  6. alert ("you clicked anchor"+i);//给出点击反应
  7. }
  8. }
  9. }
  10. window.onload = newLoad; //把newload事件赋值给页面加载
  11. </script></span>

前台代码:

[html] view plaincopyprint?

  1. <span style="font-family:KaiTi_GB2312;font-size:18px;"><body>
  2. <a id="anchor1" href="#">anchor1</a><br/>
  3. <a id="anchor2" href="#">anchor2</a><br/>
  4. <a id="anchor3" href="#">anchor3</a><br/>
  5. </body></span>

运行结果:无论点击那个anchor,总会弹出anchor4,而我们根本就没有anchor4:

当我们加载页面时,javascript中的newLoad函数已经运行完毕,由其中的循环可知,anchor已经赋值为4。但是由以前的编程经验来看,局部变量使用完毕就会销毁,但是anchor却没有,显然这里 JavaScript 采用了另外的方式。

闭包在 JavaScript 其实就是一个函数,在函数运行期被创建的,当上面的 函数被执行的时候,会创建一个闭包,而这个闭包会引用newLoad 作用域中的anchor。下面就来看看 JavaScript 是如何来实现闭包的:当执行 newLoad 函数的时候, JavaScript 引擎会创建newLoad函数执行上下文的作用域链,这个作用域链包含了 newLoad执行时的活动对象,同时 JavaScript 引擎也会创建一个闭包,而闭包的作用域链也会引用 newload的活动对象,这样当 newload执行完的时候,虽然其执行上下文和活动对象都已经释放了anchor,但是闭包还是引用着 newload 的活动对象,所以点击显示的是“you clicked anchor4”。运行期如图:

闭包优化

既然闭包出现了我们不想看到的结果,我们需要优化它。优化后的javascript(其他不变):

[javascript] view plaincopyprint?

  1. <span style="font-family:KaiTi_GB2312;font-size:18px;">     <script type="text/javascript">
  2. function newLoad() { //新建页面加载的事件
  3. for (var i = 1; i <= 3; i++) {
  4. var anchor = document.getElementById("anchor" + i); //找到每个anchor
  5. listener(anchor,i);//listener函数
  6. }
  7. }
  8. function listener(anchor, i) {
  9. anchor.onclick = function () {//为anchor添加单击事件
  10. alert("you clicked anchor" + i); //给出点击反应
  11. }
  12. }
  13. window.onload = newLoad; //把newload事件赋值给页面加载
  14. </script></span>

运行结果:提示对应的提示信息

结果分析:优化后的结果是因为,我们把声明的变量和单击事件相分离。用以上解释的作用域链解释:页面加载,先执行listener函数,而listener函数需要anchor变量,在listener函数作用域链中没有,会进入下一级作用域链,即查找newLoad中的anchor,因为在listener中已经确定了哪个anchor单击对应哪个提示信息,所以只是在newload中查找对应的anchor而已,不会等循环完毕产生anchor4。

因为接触javascript时间尚短,理解有误的地方欢迎指正。

1:【闭包在 JavaScript 其实就是一个函数,在函数运行期被创建的,当上面的 函数被执行的时候,会创建一个闭包,而这个闭包会引用newLoad 作用域中的anchor。】
应该是:当上面的 函数被执行的时候,会创建一个闭包,而这个闭包会引用newLoad 作用域中的i
2:【结果分析:优化后的结果是因为,我们把声明的变量和单击事件相分离。用以上解释的作用域链解释:页面加载,先执行listener函数,而listener函数需要anchor变量,在listener函数作用域链中没有,会进入下一级作用域链,即查找newLoad中的anchor,因为在listener中已经确定了哪个anchor单击对应哪个提示信息,所以只是在newload中查找对应的anchor而已,不会等循环完毕产生anchor4。】应该是:listener(anchor,i);函数3次执行会产生3个不同的闭包,分别引用不同的anchor 和i。所以执行 anchor.onclick = function () { alert("you clicked anchor" + i);} 时会根据3个不同的"执行上下文"的作用域链而找到不同的i变量。
3:总结:子函数会根据拥有当初时刻"执行上下文对象"的作用域链的闭包而找到当初时刻的存储的一系类值。
4:闭包算是执行时动态的分配并保持对当时状态的记忆存储。
5:函数每次执行时对应的运行期上下文都是独一无二的,所以多次调用同一个函数就会导致创建多个运行期上下文,当函数执行完毕,执行上下文会被销毁。每一个运行期上下文都和一个作用域链关联。

时间: 2024-10-25 11:51:12

JavaScript中的作用域链和闭包的相关文章

JavaScript中的作用域链原理

执行环境 作用域链的形成与执行环境(Execution Environment)相关,在JavaScript当中,产生执行环境有如下3中情形: 1 进入全局环境 2 调用eval函数 3 调用function 在一个执行环境A上可以创建执行环境B,执行环境B又可以创建执行环境C...,这一系列的执行环境构成执行环境栈,最新创建的执行环境位于栈顶(栈底永远是全局执行环境),当栈顶执行环境结束之后(与之相关的代码执行结束)就会被弹出站外,底下的执行环境就会成为新的栈顶.如下图所示: 一个执行环境由3

简单说说Javascript中的作用域链

    Javascript中作用域就是变量与函数的可访问范围,即作用域控制着变量与函数的可见性和生命周期.变量的作用域有全局作用域和局部作用域两种.当查找变量的时候,会先从当前上下文的变量对象中查找,如果没有找到,就会从父级执行上下文的变量对象中查找,一直找到全局上下文的变量对象,也就是全局对象.这样由多个执行上下文的变量对象构成的链表就叫做作用域链. 看几个题目: A : 1 var a = 1 2 function fn1(){ 3 function fn2(){ 4 console.lo

从作用域链谈闭包

闭包(closure)是Javascript语言的一个难点,也是它的特色.非常多高级应用都要依靠闭包实现. 神马是闭包 关于闭包的概念,是婆说婆有理. 因而,我就翻阅了红皮书(p178)上对于闭包的陈述: 闭包是指有权訪问另外一个函数作用域中的变量的函数 这概念有点绕,拆分一下.从概念上说,闭包有两个特点: 1.函数 2.能訪问另外一个函数作用域中的变量 在ES 6之前,Javascript仅仅有函数作用域的概念.没有块级作用域(但catch捕获的异常 仅仅能在catch块中訪问)的概念(IIF

JavaScript中的作用域 、作用域链和闭包

JavaScript中作用,作用域链和闭包详解 一.作用域在js中有全局变量和局部变量之分:比如var a = 1;function sum(){var b=1console.log(b) //1console.log(a) //2 }sum()console.log(a) //3console.log(b) //4 例子中 a 是全局变量,b是局部变量(定义在函数内部,只能在函数内部访问)所以第1行正确 函数内部也能访问全局变量 a所以第2行也能正确 第三行也正确.第4行有外部不能访问内部变量

Javascript的作用域、作用域链以及闭包

一.javascript中的作用域 ①全局变量-函数体外部进行声明 ②局部变量-函数体内部进行声明 1)函数级作用域 javascript语言中局部变量不同于C#.Java等高级语言,在这些高级语言内部,采用的块级作用域中会声明新的变量,这些变量不会影响到外部作用域. 而javascript则采用的是函数级作用域,也就是说js创建作用域的单位是函数. 例如: 在C#当中我们写如下代码: static void Main(string[] args) { for (var x = 1; x < 1

个人理解的javascript作用域链与闭包

闭包引入的前提个人理解是为从外部读取局部变量,正常情况下,这是办不到的.简单的闭包举例如下: 1 function f1(){ 2 3 n=100; 4 5 function f2(){ 6 alert(n); 7 } 8 9 return f2; 10 11 } 12 13 var result=f1(); 14 15 result(); // 100 代码中的f2函数,就是闭包. 1 function f1(){ 2 3 var n=100; 4 5 nAdd=function(){n+=1

在chrome开发者工具中观察函数调用栈、作用域链与闭包

在chrome开发者工具中观察函数调用栈.作用域链与闭包 在chrome的开发者工具中,通过断点调试,我们能够非常方便的一步一步的观察JavaScript的执行过程,直观感知函数调用栈,作用域链,变量对象,闭包,this等关键信息的变化.因此,断点调试对于快速定位代码错误,快速了解代码的执行过程有着非常重要的作用,这也是我们前端开发者必不可少的一个高级技能. 当然如果你对JavaScript的这些基础概念[执行上下文,变量对象,闭包,this等]了解还不够的话,想要透彻掌握断点调试可能会有一些困

JS详细图解作用域链与闭包

JS详细图解作用域链与闭包 攻克闭包难题 初学JavaScript的时候,我在学习闭包上,走了很多弯路.而这次重新回过头来对基础知识进行梳理,要讲清楚闭包,也是一个非常大的挑战. 闭包有多重要?如果你是初入前端的朋友,我没有办法直观的告诉你闭包在实际开发中的无处不在,但是我可以告诉你,前端面试,必问闭包.面试官们常常用对闭包的了解程度来判定面试者的基础水平,保守估计,10个前端面试者,至少5个都死在闭包上. 可是为什么,闭包如此重要,还是有那么多人没有搞清楚呢?是因为大家不愿意学习吗?还真不是,

理解js中的作用域以及初探闭包

前言 对于js中的闭包,无论是老司机还是小白,我想,见得不能再多了,然而有时三言两语却很难说得明白,反正在我初学时是这样的,脑子里虽有概念,但是却道不出个所以然来,在面试中经常会被用来吊自己的胃口,考察基础,虽然网上自己也看过不少相关闭包的文章,帖子,但貌似这玩意,越看越复杂,满满逼格高,生涉难懂的专业词汇常常把自己带到沟里去了,越看越迷糊,其实终归结底,用杨绛先生的一句话就是:"你的问题在于代码写得太少,书读得不够多",其实在我看来前者是主要的,是后者的检验, 自知目标搬砖20年(还