在全局作用域中声明变量加 var 关键字和不加 var ,js 引擎都会将这个变量声明为全局变量,在实际代码运行时,两种声明方式的变量的行为也几乎是一致的。但是在全局作用域下是否声明一个变量的时候加 var 和不加 var,js 引擎具体执行了哪些操作呢,其效果又是否完全一致?

首先我们看在一个函数体内(局部作用域)声明变量,如下:

// 变量声明不加 var
function foo (a) {
console.log(a + b) // b is not defined
b = a
} foo(2)

【分析】执行 foo(2) 的时候,我们具体看 foo 函数,首先打印了 a + b 的值,然后声明了一个 b 变量(没有使用var关键字),并将传入的 a 赋值给 b,因为 js 引擎按照代码顺序编译和执行代码,因此在打印 a + b 的时候,在任何作用域中都是无法找到 b 变量的。

在执行 foo(2) 表达式的时候 js 引擎具体的的操作过程如下:

  1. 在当前作用域中查找名为 foo 的函数(RHS)
  2. 进入 foo函数体,首先 JS 引擎在执行前会对整个脚本文件的声明部分做完整分析(包括局部变量),从而确定变量的作用域(js引擎读取一段js代码,首先执行预解析,就是逐行读取js代码,寻找全局变量和全局函数,遇到全局变量,把变量的值变为undefind,存在内存中,遇到全局函数,直接存在内存中,这个过程如果发现语法错误,预解析终止)。因此第一步搜集变量,发现在函数作用域中这里只有作为参数的局部变量 a,提升到作用域顶部
  3. 将 2 赋值给参数变量 a(a = 2, LHS)
  4. 查找 console 对象(RHS),发现是内置函数,在 console 对象下查找 log 函数(RHS)
  5. 在当前作用域中查找变量 a,并获取 a 的值为(a = 2, RHS)
  6. 在当前作用域中查找变量 b,未找到该变量
  7. 将 a 和 b 的查找结果传入 console.log() 函数,打印结果( b 未定义,抛出错误: b is not defined)
  8. 继续执行 b = a。首先获取变量 a 的值(a = 2, LHS), 然后在当前作用域中查找变量 b(RHS),未找到,到上一层作用域(全局作用域)中查找(RHS),未找到,
  9. 在全局作用作用域中创建一个名称为 b 的变量,并将其返回给引擎(注意:严格模式下禁止自动或隐式地创建全局变量)
  10. 将 a 的值(2)赋值给全局变量 b( b = a, LHS)

再看第二个例子

// 变量声明加 var
function foo (a) {
console.log(b) // undefined
console.log(a + b) // NaN
var b = a
} foo(2)

【分析】执行 foo(2) 的时候,我们看 js 引擎具体做了哪些操作?

  1. 在当前作用域中查找名为 foo 的函数(RHS)
  2. 进入 foo 函数体,搜集变量,发现声明了局部变量 a 和 b,因此将 a 和 b 提升到函数作用域的顶部(此时 b 的值为 undefined)
  3. 参数赋值,将 2 赋值给变量 a(a = 2, LHS)
  4. 查找 console 对象(RHS),发现是内置函数,在 console 对象下查找 log 函数(RHS)
  5. 在当前作用域中查找变量 b(RHS),发现已经声明,但是值为 undefined ,传给log()函数,执行打印,输出结果(b is not defined)
  6. 重复第4步 RHS 查找 console.log() 函数,查找变量 a 的值(a = 2, RHS);查找变量b的值(b = undefined, RHS),将 a 和 b 的值传入 console.log(),执行运算,输出结果(2 + undefined,结果 NaN)

通过上述在函数体内声明变量的例子,已经可以看出来 js 引擎在处理这两种情况的区别,在全局作用域中的也是如此,而且理解起来更为简单。

看两个例子:

console.log(a) // undefined
a = 3

声明变量 a 的时候没有加 var,因此 js 引擎默认将变量 a 声明为全局变量(值为 undefined)并提升到作用域顶部(为什么在console.log() 中可以访问到 a),但是此时的赋值操作需要等到 console.log() 方法执行完之后才会执行,因此在 console.log(a) 打印的结果会是 undefined

console.log(a) // undefined
var a = 3

这里的输出结果仍旧是 undefined,但是和上面的例子不同的是, js 引擎并没有主动的去创建变量 a,而是直接将变量 a 搜集到全局变量的集合中,并将 a 提升到作用域顶部。

【延伸】全局变量是 window 的属性(浏览器环境下),因此声明全局变量时是否加 var,可以通过 Object 提供的 getOwnPropertyDescriptot(object, propertyName) 来进行比较是否存在不同:

var a = 1
b = 2
console.log(Object.getOwnPropertyDescriptor(window, a))
// { value: 1, writable: true, enumerable: true, configurable: false } console.log(Object.getOwnPropertyDescriptor(window, b))
// { value: 2, writable: true, enumerable: true, configurable: true

通过结果的比较可以发现,未使用 var 声明的全局变量的configurable 属性是 true,也就是说,未通过 var 声明的变量是可以删除的,如下:

delete a
// false delete b
// true

*关于 Object.getOwnPropertyDescriptor(object, propertyName) (propertyName 需要传入字符串形式的属性名)

【参考:深入理解javascript对象系列第三篇——神秘的属性描述符

用于查询一个属性的描述符,并以对象的形式返回,返回结果的属性如下:

  • Configurable:是否可以使用 delete 删除属性,以及是否可以修改属性描述符的特性,默认值为 true
  • Enumerable:是否出现在对象的属性枚举中,比如是否可以通过 for-in 循环返回该属性,默认值为 true
  • Writable:是否可以修改属性的值,默认值为 true
  • Value:属性的数据值,读取属性值的时候,从这个位置读;写入属性值的时候,把新值保存在这个位置。默认值为 undefined

【补充知识点】

LHS 和 RHS

全局变量和局部变量

作用域、作用域链、预解析

更多变量声明是否加 var 的区别

详解变量声明加 var 和不加 var 的区别的更多相关文章

  1. javascript 中加’var‘和不加'var'的区别,你真的懂吗?

    没看之前千万别说我是标题党,这个问题真的有好多淫都不懂!!! 大家都看了很多文章,都说避免隐式声明全局变量,就是说声明变量前必须加'var',那加了'var'和不加'var'到底有啥区别呢? 先来看一 ...

  2. javascript中加var和不加var的区别

    Javascript是遵循ECMAScript标准下的一个产物,自然ECMAScript的标准其要遵循. 先来看下var关键字的定义和用法 var 语句用于声明变量. JavaScript 变量的创建 ...

  3. js中加“var”和不加“var”的区别

    JavaScript 拥有动态类型.这意味着相同的变量可用作不同的类型: var x // x 为 undefined var x = 6; // x 为数字 var x = "Bill&q ...

  4. js中加“var”和不加“var”的区别,看完觉得这么多年js白学了

    Javascript声明变量的时候,虽然用var关键字声明和不用关键字声明,很多时候运行并没有问题,但是这两种方式还是有区别的.可以正常运行的代码并不代表是合适的代码. var num = 1: 是在 ...

  5. C++中创建对象的时候加括号和不加括号的区别

    c++创建对象的语法有----- 1 在栈上创建 MyClass a; 2 在堆上创建加括号 MyClass *a= new MyClass(); 3 不加括号 MyClass *a = new My ...

  6. Java中主类中定义方法加static和不加static的区别

     Java中主类中定义方法加static和不加static的区别(前者可以省略类名直接在主方法调用(类名.方法),后者必须先实例化后用实例调用) 知识点:1.Getter and Setter 的应用 ...

  7. onclick时间加return和不加return的区别

    JAVASCRIPT在事件中调用函数时用return返回值实际上是对window.event.returnvalue进行设置. 而该值决定了当前操作是否继续.当返回的是true时,将继续操作.当返回是 ...

  8. C++中创建对象的时候加括号和不加括号的区别(转)

    c++创建对象的语法有----- 1 在栈上创建 MyClass a; 2 在堆上创建加括号 MyClass *a= new MyClass(); 3 不加括号 MyClass *a = new My ...

  9. 【转】new对象时,类名后加括号和不加括号的区别

    请看测试代码: #include <iostream> using namespace std; // 空类 class empty { }; // 一个默认构造函数,一个自定义构造函数 ...

随机推荐

  1. Python IDLE快捷键一览

    编辑状态时:Ctrl + [ .Ctrl + ] 缩进代码Alt+3 Alt+4 注释.取消注释代码行Alt+5 Alt+6 切换缩进方式 空格<=>TabAlt+/ 单词完成,只要文中出 ...

  2. Java课程设计——象棋(201521123042 姚佳希)

    1. 团队课程设计博客链接 Java课程设计(团队版) 2 个人负责模块或任务说明 ChessBoard类创建棋盘及界面. ChessPoint类创建棋盘格点及界面. ChessPiece类创建棋子及 ...

  3. 201521123054 《Java程序设计》 第十周学习总结

    1. 本周学习总结 2. 书面作业 题目4-2 1.1 截图你的提交结果(出现学号) 1.2 4-2中finally中捕获异常需要注意什么? 无论是否抛出异常,也无论从什么地方返回,finally语句 ...

  4. Intellij idea使用Junit

    之前使用idea做Junit测试的时候,都是一个一个方法来写,然后在方法名@Test这样测试-.. 后来发现eclipse有直接把整个类的方法都可以抽取出来,自动生成Junit测试方法-于是在找Ide ...

  5. 7z命令行 极限压缩指令

    摘抄自http://www.cnblogs.com/qanholas/archive/2011/10/03/2198487.html 7za a -t7z bag.7z "/home/fil ...

  6. hadoop各个类及其作用

    1.基础包(包括工具包和安全包) 包括工具和安全包.其中,hdfs.util包含了一些HDFS实现需要的辅助数据结构:hdfs.security.token.block和hdfs.security.t ...

  7. CSS3 animation-timing-function steps()

    animation-timging-function 主要是控制css动画从开始到结束的速度. linear:线性过渡.等同于贝塞尔曲线(0.0, 0.0, 1.0, 1.0) ease:平滑过渡.等 ...

  8. GitHub使用(一) - 新建个人网站

    1.首先进入“仓库Repositories”,点击“新建New”.

  9. Kafka快速上手(2017.9官方翻译)

    为了帮助国人更好了解.上手kafka,特意翻译.修改了个文档.官方Wiki : http://kafka.apache.org/quickstart 快速开始 本教程假定您正在开始新鲜,并且没有现有的 ...

  10. Hadoop安全(1)——————美团Hadoop安全实践

    http://tech.meituan.com/hadoop-security-practice.html 前言 在2014年初,我们将线上使用的 Hadoop 1.0 集群切换到 Hadoop 2. ...