1、作用域

  作用域,它是指对某一变量和方法具有访问权限的代码空间。当我们在定义变量的时候,会定义两种变量,一种是在全局环境下定义的变量,叫全局变量,一种是在函数中定义的变量叫局部变量。全局变量的作用域就是全局环境,局部变量的作用域就是函数。

2、作用域链

  javascript高级程序设计书中写到,当代码在一个环境中执行时,会创建变量对象的一个作用域链。作用域链的用途,是保证对执行环境有权访问的所有变量和函数的有序访问。作用域链的前端,始终都是当前执行的代码所在环境的变量对象。 作用域链中的下一个变量对象来自包含(外部)环境,而再下一个变量则来自下一个包含环境,这样一直延续到全局执行环境;全局执行环境的变量对象始终都是作用域链中的最后一个对象。

  简单来说就是,当我查找一个变量时,我会先在自己所在的环境找,找不到延着作用域链继续向外一层的环境找,找不到再向外找,一直找到全局执行环境中。

  看一小段代码:

  

 var colorA = "red";
function abc(){
var colorB = "blue";
function info(){
var color ="yellow";
console.log(colorA);-------1
console.log(colorB);-------1
console.log(colorC);-------1
};
info();
console.log(colorA);-------2
console.log(colorB);-------2
console.log(colorC);-------2
};
abc();
console.log(colorA);-------3
console.log(colorB);-------3
console.log(colorC);-------3

  在上述代码中,你可以发现,标记为1的,三个颜色A、B、C都能正确的被找到并打印出来。而标记为2的,只能找到A、B颜色。而标记为3的,只能够找到A颜色。

  出现这种情况的原因就是去查找变量值的时候,是先从自身所在的环境中查找,没找到的时候,沿着作用域链向外层查找,所以在内层可以访问到外层定义的变量,在外层是访问不到内层所定义的变量的。

3、块级作用域和函数作用域

  我们都听说过javscript是没有块级作用域的,javascript执行的是函数作用域,那么什么是块级作用域呢?

  任何一对花括号({和})中的语句集都属于一个块,在这之中定义的所有变量在代码块外都是不可见的,我们称之为块级作用域。

  什么是函数作用域?

  简单来说函数中的参数和变量在函数外部是不可见的,只在函数内部可以访问。

  为什么说javascript没有块级作用域,看一段以下代码:

  

 var color="red";
if(true){
var color = "blue";
console.log(color);
}
console.log(color);

  会发现两个输出的都是blue,如果有块级作用域的话,if里面的color会被当做局部变量创建,这样子的话输出的就是"blue","red"了;正是因为没有块级作用域,if里面对color的赋值被当做是全局变量把原来的值覆盖掉,所以输出才会是两个都是blue;

4、变量声明提升

  先来看一小段代码,你认为的输出是什么:

  

 var a = 100;
function testResult(){
var b = 2 * a;
var a = 200;
var c = a / 2;
console.log(b);
console.log(c);
}
testResult();

  我们先不管上面的输出是什么,先来看看这几行代码的输出是什么;

  console.log(a);var a = 123;  //undefined;

  console.log(a);var a ;  //undefined;

  var a = 123;console.log(a);   //123;

  出现这种情况就是因为变量声明提升,我们去声明一个变量的时候javascript引擎会把变量"提前"声明,而把赋值放在原来的顺序。

  所以console.log(a);var a = 123;的执行顺序其实是var a; console.log(a); a = 123;

  故输出a的时候只是创建了变量还没有被赋值,所以输出为undefined;

  现在再来看看刚那段代码的执行顺序其实是:

  

 var a = 100;
function testResult(){
var b;
var a;
var c;
b = 2 * a; //此时在局部变量中找到了a,故不会再向上找全局变量a = 100; 但是a还没被赋值,a为undefined,故此时b为NaN;
a = 200;
c = a / 2; // 100;
console.log(b); //NaN
console.log(c); //100
}
testResult();

注:对undefined作加减乘除会把它转化为数字类型,undefined转化为数字类型就是NaN;

故输出为 NaN 和 100 ;

  

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

  1. 深入理解 JavaScript 变量的作用域和作用域链

    一个变量的作用域(scope)是程序源代码中定义这个变量的区域.简单的说,作用域就是变量与函数的可访问范围.全局变量拥有全局作用域,在JavaScript代码中的任何地方都有定义.局部变量是在函数体内 ...

  2. 一步步学习javascript基础篇(2):作用域和作用域链

    作用域和作用域链 js的语法用法非常的灵活,且稍不注意就踩坑.这集来分析下作用域和作用域链.我们且从几道题目入手,您可以试着在心里猜想着答案. 问题一. if (true) { var str = & ...

  3. 关于Javascript作用域及作用域链的总结

    本文是根据以下文章以及<Javascript高级程序设计(第三版)>第四章相关内容总结的. 1.Javascript作用域原理,地址:http://www.laruence.com/200 ...

  4. javascript篇-----函数作用域,函数作用域链和声明提前

    在一些类似C语言的编程语言中,花括号内的每一段代码都具有各自的作用域,而且变量在声明它们的代码段之外是不可见的(也就是我们不能在代码段外直接访问代码段内声明的变量),我们称之为块级作用域,然而,不同于 ...

  5. javascript笔记:javascript的关键所在---作用域链

    javascript里的作用域是理解javascript语言的关键所在,正确使用作用域原理才能写出高效的javascript代码,很多javascript技巧也是围绕作用域进行的,今天我要总结一下关于 ...

  6. 浅谈JavaScript中的变量、参数、作用域和作用域链

    基本类型和引用类型 在JavaScript中有两种数据类型值.基本类型值和引用类型值.基本类型值指的是简单的数据段,而引用类型值指的是可能由多个值构成的对象.在JavaScript中有5种基本数据类型 ...

  7. javascript作用域和作用域链摘录

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

  8. JavaScript高级之词法作用域和作用域链

    主要内容: 分析JavaScript的词法作用域的含义 解析变量的作用域链 变量名提升时什么 一.关于块级作用域         说到JavaScript的变量作用域,与咱们平时使用的类C语言不同. ...

  9. JavaScript 开发进阶:理解 JavaScript 作用域和作用域链

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

  10. JavaScript 开发进阶:理解 JavaScript 作用域和作用域链(转载 学习中。。。)

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

随机推荐

  1. dedecms文章的更新时间问题 每次更改文章时间变成最新的

    dedecms 每次更改文章,更新时间这里每次改了后再来看又变成当前最新时间的了. 解决方法: 查找后台目录的 templets/article_edit.htm 这个文件. 然后打开,查找如下代码: ...

  2. 解决gradle:download特别慢的问题

    使用AndroidStudio 2.2.2 新增加了一个dependencies,需要下载jar包,此时就会卡在 gradle:download https://….. 这个状态中. 原因就是因为我们 ...

  3. 为GCD队列绑定NSObject类型上下文数据-利用__bridge_retained(transfer)转移内存管理权-备

    下面评论的好友“@Jim”给了种新的思路,就是在清除context的函数里面,用“_bridge_transfer”转换context,把context的内存管理权限重新交给ARC,这样,就不用显式调 ...

  4. zlog使用手册,小靠谱啊

    http://hardysimpson.github.io/zlog/UsersGuide-CN.html Chapter 1 zlog是什么? zlog是一个高可靠性.高性能.线程安全.灵活.概念清 ...

  5. Python字符串处理NoneType的处理

    Python合并处理字符串的四种方法在这里都有介绍: http://www.cnblogs.com/heshizhu/archive/2012/01/11/2319892.html 无论使用最简单的+ ...

  6. ubuntu 终端只显示当前目录名称

    修改.bashrc文件: 原来: #修改终端提示颜色 color_prompt=yes if [ "$color_prompt" = yes ]; then PS1='${debi ...

  7. 关于API的设计和需求抽象

    一,先来谈抽象吧,因为抽象跟后面的API的设计是息息相关的 有句话说的好(不知道谁说的了):计算机科学中的任何问题都可以抽象出一个中间层就解决了. 抽象是指在思维中对同类事物去除其现象的.次要的方面, ...

  8. smarty 从配置文件读取变量

    smarty变量分3种: Variables [变量] Variables assigned from PHP [从PHP分配的变量] Variables loaded from config fil ...

  9. 【转】彻底解决INSTALL_FAILED_UPDATE_INCOMPATIBLE的安装错误

    原文网址:http://bbs.9ria.com/thread-245162-1-1.html 利用adb shell进入系统,进入/data/app或者/data/data,删除跟你安装的apk同样 ...

  10. 段错误bug的调试

    我们在用C/C++语言写程序的时侯,内存管理的绝大部分工作都是需要我们来做的.实际上,内存管理是一个比较繁琐的工作,无论你多高明,经验多丰富,难 免会在此处犯些小错误,而通常这些错误又是那么的浅显而易 ...