javaScript 变量提升 var let const,以及JS 的解析阶段和执行阶段
我们先来看一道面试题,大家猜想一下,下面这段代码,打印出来的结果是什么
var name = 'World!';
(function () {
if (typeof name === 'undefined') {
var name = 'Jack';
console.log('Goodbye ' + name);
} else {
console.log('Hello ' + name);
}
})();
这里打印出来的结果为什么是 Goodbye Jack, 而不是Hello World呢。
因为JS代码运行分为两个阶段:解析阶段和执行阶段
解析阶段:找到所有的声明,包括函数声明和var声明,并把声明操作提升到它所在执行环境的顶部(全局或者函数内),而赋值和逻辑操作则会被留在原地等待代码执行。
并且,当它发现了函数名与函数名或者函数名与变量名冲突的时候,处理原则是:‘处理函数声明冲突时会覆盖,处理变量声明时冲突会忽略’。(后面再详解)
经过解析阶段之后,我们的代码其实被解析成了这样:
var name = 'World!';
(function () {
var name; // 对变量name的声明会被提升到函数的顶部,并且局部的变量name会覆盖全局的变量name
if (typeof name === 'undefined') {
name = 'Jack';
console.log('Goodbye ' + name);
} else {
console.log('Hello ' + name);
}
})();
执行阶段:赋值和逻辑操作就会被放在执行阶段。所以当我们在执行阶段时,由于name变量的声明被提升了,在if内判断typeof name的时候,赋值操作(即name='Jack')还未执行,所以 typeof name === 'undefined' 判断结果为true,打印出来的结果就是‘GoodBye Jack’。
而在ES6中新规范中的 let,const 会不会执行变量提升呢? 我做了这么一个小实验
(function () {
console.log(i); // undefined
var i = 'test const';
})();
(function () {
console.log(i); // ReferenceError: i is not defined
let i = 0;
})();
(function () {
console.log(i); // ReferenceError: i is not defined
const i = 'test const';
})();
可以看到 let 和const 声明的变量是不会被提升的,这也是为什么一开始的时候我把字体标红:函数声明和var声明会被提升。
那我们接着说一下函数声明的提升。
定义函数有两种非常常见的方式
function A(){ // 函数声明
alert('A');
} var B = function(){ // 函数表达式
alert('B')
}
对于用函数声明的方式定义的函数,我们可以在函数之前调用,这就是因为JS在解析阶段把整个函数提升到了全局环境的最上面,
而对于函数表达式方式定义的函数,我们在函数之前调用的话,会得到undefined的结果,因为var 声明的变量被提升了,但是赋值操作还没有执行
我们看一下执行结果
console.log(A); // ƒ A(){ alert('A'); }
function A(){
alert('A');
} console.log(B); // undefined
var B = function(){
alert('B')
}
关于js的解析和执行阶段的补充详解,将在下一篇博客中讲解
javaScript 变量提升 var let const,以及JS 的解析阶段和执行阶段的更多相关文章
- JavaScript变量声明var,let.const
var声明变量的作用域限制在其声明位置的上下文中 var x = 0; // x是全局变量,并且赋值为0. console.log(typeof z); // undefined,因为z还不存在. f ...
- javascript变量提升详解
js变量提升 对于大多数js开发者来说,变量提升可以说是一个非常常见的问题,但是可能很多人对其不是特别的了解.所以在此,我想来讲一讲. 先从一个简单的例子来入门: a = 2; var a; cons ...
- 回归基础: JavaScript 变量提升
from me: javascript的变量声明具有hoisting机制,它是JavaScript一个基础的知识点,也是一个比较容易犯错的点,平时在开发中,大大小小的项目都会遇到. 它是JavaScr ...
- JavaScript变量提升 面试题
<!DOCTYPE html> <html lang="zh"> <head> <meta charset="UTF-8&quo ...
- JavaScript变量提升和函数声明预解析
1.首先理解函数作用域 在JavaScript中,变量的定义并不是以代码块作为作用域的,而是以函数作用作用域的.也就是说,如果变量是在某个函数中定义的,那么它在函数以外的地方是不可见的.而如果该变量是 ...
- var 声明变量的变量提升问题(let/const)
在ES6之前,JavaScript没有块级作用域(一对花括号{}即为一个块级作用域),只有全局作用域和函数作用域.变量提升即将变量声明提升到它所在作用域的最开始的部分.既全局变量. <scrip ...
- js javascript变量提升
var:变量提升(无论声明在何处,都会被提至其所在作用域的顶部) let:无变量提升(所在的块内,未到let声明时(即let声明之前),是无法访问该变量的(not defined)),let变量不能重 ...
- JavaScript: 变量提升和函数提升
第一篇文章中提到了变量的提升,所以今天就来介绍一下变量提升和函数提升.这个知识点可谓是老生常谈了,不过其中有些细节方面博主很想借此机会,好好总结一下. 今天主要介绍以下几点: 1. 变量提升 2. 函 ...
- javascript变量提升
提升是一种将变量和函数的声明移到函数作用域(如果不在任何函数内的话就是全局作用域)最顶部的机制. 提升影响了变量的生命周期,一个变量的生命周期包含3个阶段: 声明——创建一个新变量,例如var myV ...
随机推荐
- ArrayList初始化的4种方法
In the last post we discussed about class ArrayList in Javaand it’s important methods. Here we are s ...
- linux下通过命令行上传文件到百度网盘
一.环境: centos release 6.9 python 2.7.13 二.安装工具bypy sudo pip install bypy 三.使用bypy 3.1 授权 [root@ineedl ...
- linux下设置软件使用socks5代理
1.为wget使用代理,可以直接修改/etc/wgetrc,也可以在主文件夹下新建.wgetrc,并编辑相应内容,本文采用后者. 直接往~/.wgetrc(自行创建此文件)添加如下内容: https_ ...
- B-树 B+树 B*树
区分B树,B-树 有的文章说二叉查找树(Binary Search Tree,BST)就是B树,这个我总结来说是不对的 B树和B-树是同一种树,只不过英语中B-tree被中国人翻译成了B-树,让人以为 ...
- 导出数据库表为world文档说明,以及PowerDesigner导出表结构pdm设计文档
如何使用“mysql导出数据库结构为world工具”以及如何使用powerdesigner映射数据库模型 一.通过powerdesigner配置ojdbc 1.安装并打开powerdesigner,新 ...
- 如何解决Visual Studio2010 编译时提示系统找不到指定文件问题
前一段时间,开始使用vs2010编写程序,可是在编译的时候总是报错,提示系统找不到指定文件,导致无法正常运行程序,花了好久时间终于找到原因解决,是因为常规的输出目录 要与链接的常规的输出文件要相对应. ...
- Mac OSX 安装qemu
参考: Installing QEMU on OS X Homebrew Mac OSX 安装qemu 1.Install Homebrew: /usr/bin/ruby -e "$(cur ...
- UVa 10118 免费糖果(记忆化搜索+哈希)
https://vjudge.net/problem/UVA-10118 题意: 桌上有4堆糖果,每堆有N颗.佳佳有一个最多可以装5颗糖的小篮子.他每次选择一堆糖果,把最顶上的一颗拿到篮子里.如果篮子 ...
- Python subprocess模块学习总结--转载
一.subprocess以及常用的封装函数运行python的时候,我们都是在创建并运行一个进程.像Linux进程那样,一个进程可以fork一个子进程,并让这个子进程exec另外一个程序.在Python ...
- 【Django】【四】测试
[Testing in Django] 通过参数可控制Django项目不同级别的测试. 1. 运行sign应用下所有的测试用例: \\guest\python manage.py test sign ...