ES6新增了两个重要的关键字let
和const
,相信你们都不陌生,可是包括我在内,在系统学习ES6以前也只使用到了【不存在变量提高】这个特性。javascript
var
定义的变量会提高到整个函数做用域内,let/const
则支持块级做用域。java
块级做用域: 由
{}
包裹的做用域(函数那种{}不算)
来看一个var
的例子:面试
{ var a = 1; } console.log(a);
此时输出1,由于var
没有块级做用域。函数
来看一个let
的例子(const
效果同样):学习
{ let a = 1; } console.log(a);
此时会报错ReferenceError
,由于let/const
支持块级做用域,因此let
定义的a
只在{}
能够访问spa
与var
不一样的是,let/const
声明的变量不存在变量提高,也就是说{}
对于let/const
是有效的。code
来看一个var
的例子:ip
console.log(a); var a = 1;
此时会输出undefined,由于var声明的变量会提高到做用域顶部(只提高声明,不提高赋值)作用域
来看一个let
的例子(const
效果也同样):it
console.log(a); let a = 1;
此时会报错ReferenceError
,由于let
不存在变量提高
同一做用域内let/const
不能够重复声明,var
能够。
来看一个var
的例子:
var a = 1; var a = 2; console.log(a);
此时会输出2,var是支持重复声明的,后面声明的值会覆盖前面声明的值。
来看一个let
的例子(const
效果也同样):
let a = 1; let a = 2; console.log(a);
此时会报错SyntaxError
,由于同一做用域内let/const
不能够重复声明。
再来看一个不一样做用域的例子:
let a = 1; { let a = 2; } console.log(a);
此时输出1,由于二者做用域不一样
暂存死区TDZ(Temporal Dead Zone)是ES6中对做用域新的语义。
经过let/const
定义的变量直到执行他们的初始化代码时才被初始化。在初始化以前访问该变量
会致使ReferenceError
。该变量处于一个自做用域顶部到初始化代码
之间的“暂存死区”中。
来看如下例子:
function do_something() { console.log(bar); // undefined console.log(foo); // ReferenceError var bar = 1; let foo = 2; } do_something();
var
定义的变量声明会提高到做用域顶部,因此bar
是undefined,而let
定义的变量从做用域开始到let foo=2
这中间都没法访问,访问会报错ReferenceError
typeof检测var定义的变量或者检测不存在的变量时会返回undefined,若是检测暂存死区内的变量,会报错ReferenceError
.
console.log(typeof foo); // undefined console.log(typeof bar); // ReferenceError console.log(typeof bar2); // undefined let bar = 1; var bar2 = 2;
也就是说typeof去检测未初始化的let
变量时会报错,var
或者未声明的变量不会报错
function test(){ var foo = 33; { let foo = (foo + 55); } } test();
以上函数执行结果是什么?为何?
报错
{}
内有let
定义的foo
,因此存在暂存死区,(foo + 55)
这个表达式是在let foo
以前执行的(赋值时先执行等号右边的,执行完毕把结果赋给等号左边),表达式执行的时候尚未初始化foo,因此报错ReferenceError
let b = 1; function test4() { console.log(b); let b = 2; } test4()