每个函数都有自己的作用域,当执行流进入一个函数时,函数就会被推入栈中,而在函数执行之后,栈将其执行环境弹出,把控制权放回给之前的作用域,全局作用域是最外围的一个作用域,因此,所有全局变量和函数都是作为window对象的属性和方法创建的。在某个方法函数的作用域中,所有代码执行完之后,该作用域被销毁,保存在其中的所有变量和函数定义也会随着被销毁,这就是局部作用域。

(PS:全局作用域直到应用程序退出,例如关闭网页活浏览器,才会被销毁。)

我个人理解的作用域链就是,当你声明一个函数时,局部作用域一级一级往上扣起来,就是作用域链。

如图:

作用域链的用途,是保证对执行环境有权访问的所有变量和函数的有序访问。

好了好了~  书面语言终于说完了。我们还是来看看代码吧~

先来个基础点的:

		var color = "blue";
function changeColor(){
  var anothercolor = "red";
if(color==="blue"){
color = anothercolor;
}
  //这里可以访问anothercolor,color }
//这里只可以访问color
changeColor();
console.log(color);//red
console.log(anothercolor);// undefined

  解析:函数changeColor()的作用域链包含两个对象,它自己的变量对象和全局环境的变量对象。全局环境的变量对象(即window的对象)有color,changColor(),而changeColor()的变量对象是color,anothercolor。

(用我自己的话来解释就是:父级不能访问子级的变量,但是子级可以父级的变量,祖父的变量,曾祖父的变量......哈哈~。并且我们执行流的访问顺序也是从子级开始,一级一级往上查找你的需要的变量,最后一级就是全局变量。)

来看一下作用域链的典型栗子:

var x = 10;

	function foo(){

		var y = 20;

		function bar(){
var z = 30; console.log(x+y+z);
}; bar(); }; foo();//60

  解析:上面代码的输出结果是“60”,函数bar()可以直接访问"z",然后通过作用域链访问上层的“x”和“y”。

大概基本概念弄清楚了吧?!~~

那我要开始讲一些注意事情。。

JS没有块级作用域。

说明:在其他类C语言中,由{花括号}封闭的代码块都有自己的作用域,但是随和亲切的JS并非如此。

举个栗子:

if(true){
var color = "blue";
}
alert(color);//blue

  解析:如果在C,C++,或JAVA中,color会在if语句执行完毕被销毁。但在JS中,if语句中的变量声明会将变量添加到当前的作用域(这里的全局环境)。

四不四觉得有点郁闷?怎么color这个变量还存在,不应该执行完之后,被销毁了吗?

一开始小编也是很郁闷,再想想JS的执行流,小编目前的理解是:JS一切皆对象,只有函数方法可以封装,也只有函数的执行需要被调用,所以每个函数方法都有自己的作用域,而像if和for循环等等,这些直接执行的引用函数,所定义的变量都会被添加在与它同兄弟级的作用域中。

现在再来看个栗子:

for(var i = 0 ; i < 10;i++){
doSomething(i);
}
alert(i);//10

解析:由for语句创建的变量i即使在for循环执行结束后,也依旧会存在于循环外部的执行环境中。

 结合作用域看闭包


在JS中,闭包和作用域有紧密的关系。相信大家对下面的闭包栗子一定灰常熟悉,代码中通过闭包实现了一个简单的计算器。

function counter(){
var x = 0; return{
increase:function increase(){return ++x;},
decrease:function decrease(){return --x;}
};
} var ctor = counter(); console.log(ctor.increase());//1
console.log(ctor.decrease());//0

  解析:四不四又纳闷了,怎么counter()函数退出了执行上下文栈,但是变量x还没有被销毁。闭包有三大特性:1.函数嵌套函数 2.函数内部可以引用外部的参数和变量 3.参数和变量不会被垃圾回收机制回收。这里很明显,counter()函数利用闭包,把变量x引用到全局变量中。当var ctor = counter(),执行完毕后,ctor={increase:function(){...},decrease:function(){...}},这里需要注意的是,counter虽然退出了执行上下文栈,但是因为ctor中的成员仍然引用counter 的活动变量,所以counter的变量x依然在作用域中。

(附上个人对闭包的理解:在函数嵌套函数中,子函数获取父函数的私有变量,通过return引用到外部的作用域中。)

作用域链的主要作用就是用来变量查找,但是在JS中还有原型链的概念。

于是对于二维作用域链查找可以总结为:当代码需要查找一个属性或者描述符的时候,首先会通过作用域链来查找相关的对象,一旦对象被找到,就会根据对象的原型链来查找属性。

下面来举个栗子:

var foo = {}
function baz(){ Object.prototype.a = 'Set foo.a from prototype'; return function inner(){
console.log(foo.a);
}
}
baz()();//Set foo.a from prototype

  解析:对于这个栗子,流程是这样的,在inner()并没有发现foo,就通过作用域链去baz查找,也没有在baz里面找到作用域链,就去到全局环境,找到了foo,但是并没有找到属性a,于是就去到了foo._proto_的原型链中找到了属性a,便输出该值。

文章说明:个人查看各种资料,原创所得,观点不一定准确,欢迎各路大牛勘误,小女子感激不尽。

JS的作用域和作用域链的更多相关文章

  1. 了解 JS 作用域与作用域链

    (1)作用域 一个变量的作用域(scope)是程序源代码中定义的这个变量的区域. 1. 在JS中使用的是词法作用域(lexical scope) 不在任何函数内声明的变量(函数内省略var的也算全局) ...

  2. Js作用域与作用域链详解

    一直对Js的作用域有点迷糊,今天偶然读到Javascript权威指南,立马被吸引住了,写的真不错.我看的是第六版本,相当的厚,大概1000多页,Js博大精深,要熟悉精通需要大毅力大功夫. 一:函数作用 ...

  3. Js作用域与作用域链详解[转]

     一直对Js的作用域有点迷糊,今天偶然读到JavaScript权威指南,立马被吸引住了,写的真不错.我看的是第六版本,相当的厚,大概1000多页,Js博大精深,要熟悉精通需要大毅力大功夫. 一:函数作 ...

  4. js作用域与作用域链

    一直对Js的作用域有点迷糊,今天偶然读到JavaScript权威指南,立马被吸引住了,写的真不错.我看的是第六版本,相当的厚,大概1000多页,Js博大精深,要熟悉精通需要大毅力大功夫. 一:函数作用 ...

  5. js学习--变量作用域和作用域链

    作为一名菜鸟的我,每天学点的感觉还是不错的.今天学习闭包的过程中看到作用域与作用域链这两个概念,我觉得作为一名有追求的小白,有必要详细了解下. 变量的作用域 就js变量而言,有全局变量和局部变量.这里 ...

  6. JS -- The Scope Chain 作用域链

    The Scope Chain JavaScript is a lexically scoped language: the scope of a variable can be thought of ...

  7. js 作用域,作用域链,闭包

    什么是作用域? 作用域是一种规则,在代码编译阶段就确定了,规定了变量与函数的可被访问的范围.全局变量拥有全局作用域,局部变量则拥有局部作用域. js是一种没有块级作用域的语言(包括if.for等语句的 ...

  8. Js 作用域与作用域链与执行上下文不得不说的故事 ⁄(⁄ ⁄•⁄ω⁄•⁄ ⁄)⁄

    最近在研究Js,发现自己对作用域,作用域链,活动对象这几个概念,理解得不是很清楚,所以拜读了@田小计划大神的博客与其他文章,受益匪浅,写这篇随笔算是自己的读书笔记吧~. 作用域 首先明确一个概念,js ...

  9. 理解js中的作用域,作用域链以及闭包

    作用域变量作用域的类型:全局变量和局部变量全局作用域对于最外层函数定义的变量拥有全局作用域,即对任何内部函数来说,都是可以访问的 <script> var outerVar = " ...

随机推荐

  1. c#之线程入门

    C#支持通过多线程并行地执行代码,一个线程有它独立的执行路径,能够与其它的线程同时地运行.一个C#程序开始于一个单线程,这个单线程是被CLR和操作系统(也称为“主线程”)自动创建的,并具有多线程创建额 ...

  2. QT5-控件-QScrollArea-可以用于把一个窗口分割为多个-比如根据图片大小显示滚动条

    #ifndef MAINWINDOW_H #define MAINWINDOW_H #include <QMainWindow> #include <QLabel> #incl ...

  3. phpcms 2008和discuz X3.1实现同步登陆退出论坛(已实现)

    网络上文章很多,按步骤配置好了之后phpcms可以同步登录dz,但是dz登录后状态却无法同步到phpcms,网络上找了很多资料都大同小异,头大.只能自己调试了,废话不多说了.       以下网络上抄 ...

  4. js 音乐

    define(function(require,exports,module){ var $music = document.getElementById('music'); var $music_m ...

  5. PHP微信红包的算法实现探讨

    header("Content-Type: text/html;charset=utf-8");//输出不乱码,你懂的 $total=10;//红包总额 $num=8;// 分成8 ...

  6. php实现网页HTML标签补全方法

    如果你的网页内容的html标签显示不全,有些表格标签不完整而导致页面混乱,或者把你的内容之外的局部html页面给包含进去了,我们可以写个函数方法来补全html标签以及过滤掉无用的html标签. php ...

  7. c语言判断用户是否输入-非阻塞函数kbhit

    一.基础研究 要从地址读取数据,肯定是要定义一个指针变量p,用它来实现变换地址和取值的功能.另外程序是当两个条件中的某一个出现时才停止,所以应该用while~do循环语句循环输出n和d,并用while ...

  8. 委托 delegate, 继承

    c# 的委托就是说把函数当参数来传递. 这个在js完全就用不着搞什么委托东西,直接转就是了嘛.对不对!怎么录嘛! 一个函数,如果它的参数是函数,那么是这样子写的 public void method( ...

  9. 使用Mono Runtime Bundle制作安装包让C#桌面应用程序脱离net framework

    在Xamain 未被收购之前,这货monodroid.exe  就是一个打包的绑定...无奈 配置环境复杂,未能实现 ...有mono运行时就行了..不折腾了 玛德 让C#程序独立运行(脱离 .NET ...

  10. 一个坑:java.sql.ResultSet.getInt==》the column value; if the value is SQL NULL, the value returned is 0

    Retrieves the value of the designated column in the current row of this ResultSet object as a String ...