乐趣区

不只是块级作用域你不知道的let和const

ES6 新增了两个重要的关键字 letconst,相信大家都不陌生,但是包括我在内,在系统学习 ES6 之前也只使用到了【不存在变量提升】这个特性。

  • let声明一个块级作用域的本地变量
  • const语句声明一个块级作用域的本地常量,不可以重新赋值

支持块级作用域

var定义的变量会提升到整个函数作用域内,let/const则支持块级作用域。

块级作用域: 由 {} 包裹的作用域(函数那种 {} 不算)

来看一个 var 的例子:

{var a = 1;}
console.log(a);

此时输出 1,因为 var 没有块级作用域。

来看一个 let 的例子 (const 效果一样):

{let a = 1;}
console.log(a);

此时会报错 ReferenceError,因为let/const 支持块级作用域,所以 let 定义的 a 只在 {} 可以访问

不存在变量提升

var 不同的是,let/const声明的变量不存在变量提升,也就是说 {} 对于 let/const 是有效的。

来看一个 var 的例子:

console.log(a);
var a = 1;

此时会输出 undefined,因为 var 声明的变量会提升到作用域顶部(只提升声明,不提升赋值)

来看一个 let 的例子 (const 效果也一样):

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

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

总结

  1. let/const 支持函数作用域和块级作用域,var 只有函数作用域
  2. let/const 不存在变量提升,var 存在变量提升
  3. let/const 同一作用域内不可以重复声明,var 可以重复声明
  4. let/const 存在暂存死区,var 不存在

面试题

let b = 1;

function test4() {console.log(b);
    let b = 2;
}
test4()

退出移动版