首页 > 其他 > 详细

let 和 const 命令

时间:2019-12-21 12:43:23      阅读:81      评论:0      收藏:0      [点我收藏+]

let 命令

ES6 新增了let命令,用来声明变量。它的用法类似于var但是所声明的变量,只在let命令所在的代码块内有效

技术分享图片

 

 

 

上面代码在代码块之中,分别用letvar声明了两个变量。然后在代码块之外调用这两个变量,结果let声明的变量报错,

var声明的变量返回了正确的值。这表明,let声明的变量只在它所在的代码块有效。

 

不存在变量提升

var命令会发生“变量提升”现象,即变量可以在声明之前使用,

值为undefined。这种现象多多少少是有些奇怪的,按照一般的逻辑,变量应该在声明语句之后才可以使用。

为了纠正这种现象,let命令改变了语法行为,它所声明的变量一定要在声明后使用,否则报错。

技术分享图片

 

 

 

暂时性死区

 只要块级作用域内存在let命令,它所声明的变量就“绑定”(binding)这个区域,不再受外部的影响。

技术分享图片

 

 上面代码中,存在全局变量tmp,但是块级作用域内let又声明了一个局部变量tmp,导致后者绑定这个块级作用域,所以在let声明变量前,对tmp赋值会报错。

 

ES6 明确规定,如果区块中存在letconst命令,这个区块对这些命令声明的变量,从一开始就形成了封闭作用域。

凡是在声明之前就使用这些变量,就会报错

总之,在代码块内,使用let命令声明变量之前,该变量都是不可用的。这在语法上,称为“暂时性死区”(temporal dead zone,简称 TDZ)。

 

 

技术分享图片

 

 

上面代码中,在let命令声明变量tmp之前,都属于变量tmp的“死区”。

“暂时性死区”也意味着typeof不再是一个百分之百安全的操作。

 

技术分享图片

 

 

上面代码中,变量x使用let命令声明,所以在声明之前,都属于x的“死区”

只要用到该变量就会报错。

因此,typeof运行时就会抛出一个ReferenceError

作为比较,如果一个变量根本没有被声明,使用typeof反而不会报错。

 

技术分享图片

 

 

上面代码中,undeclared_variable是一个不存在的变量名,结果返回“undefined”。

所以,在没有let之前,typeof运算符是百分之百安全的,永远不会报错。

现在这一点不成立了。

这样的设计是为了让大家养成良好的编程习惯,变量一定要在声明之后使用,否则就报错。

 

 

有些“死区”比较隐蔽,不太容易发现。

技术分享图片

 

 

上面代码中,调用bar函数之所以报错(某些实现可能不报错),是因为参数x默认值等于另一个参数y

而此时y还没有声明,属于“死区”。如果y的默认值是x,就不会报错,因为此时x已经声明了

 

技术分享图片

 

 

 

另外,下面的代码也会报错,与var的行为不同。

技术分享图片

 

 

上面代码报错,也是因为暂时性死区。

使用let声明变量时,只要变量在还没有声明完成前使用,就会报错。

上面这行就属于这个情况,在变量x的声明语句还没有执行完成前,就去取x的值,导致报错”x 未定义“。

ES6 规定暂时性死区和letconst语句不出现变量提升,主要是为了减少运行时错误,防止在变量声明前就使用这个变量,从而导致意料之外的行为。

这样的错误在 ES5 是很常见的,现在有了这种规定,避免此类错误就很容易了。

总之,暂时性死区的本质就是,只要一进入当前作用域,所要使用的变量就已经存在了,但是不可获取,只有等到声明变量的那一行代码出现,才可以获取和使用该变量

 

不允许重复声明

let不允许在相同作用域内,重复声明同一个变量。

技术分享图片

 

 

因此,不能在函数内部重新声明参数。

 

技术分享图片

 

 

块级作用域

为什么需要块级作用域?

ES5 只有全局作用域和函数作用域,没有块级作用域,这带来很多不合理的场景。

第一种场景,内层变量可能会覆盖外层变量。

 

技术分享图片

 

 

上面代码的原意是,if代码块的外部使用外层的tmp变量,内部使用内层的tmp变量。

但是,函数f执行后,输出结果为undefined,原因在于变量提升,导致内层的tmp变量覆盖了外层的tmp变量。

 

第二种场景,用来计数的循环变量泄露为全局变量。

技术分享图片

 

 

上面代码中,变量i只用来控制循环,但是循环结束后,它并没有消失,泄露成了全局变量。

 

ES6 的块级作用域

let实际上为 JavaScript 新增了块级作用域。

技术分享图片

 

 上面的函数有两个代码块,都声明了变量n,运行后输出 5。

这表示外层代码块不受内层代码块的影响。如果两次都使用var定义变量n,最后输出的值才是 10。

 

ES6 允许块级作用域的任意嵌套。

技术分享图片

 

 上面代码使用了一个五层的块级作用域,每一层都是一个单独的作用域。

 第四层作用域无法读取第五层作用域的内部变量。

 

内层作用域可以定义外层作用域的同名变量。

技术分享图片

 

 

 

块级作用域的出现,实际上使得获得广泛应用的匿名立即执行函数表达式(匿名 IIFE)不再必要了

 

技术分享图片

 

 

块级作用域与函数声明

函数能不能在块级作用域之中声明?  这是一个相当令人混淆的问题。

ES5 规定,函数只能在顶层作用域和函数作用域之中声明,不能在块级作用域声明

技术分享图片

 

 

上面两种函数声明,根据 ES5 的规定都是非法的。

但是,浏览器没有遵守这个规定,为了兼容以前的旧代码,还是支持在块级作用域之中声明函数,因此上面两种情况实际都能运行,不会报错。

 

 

ES6 引入了块级作用域,明确允许在块级作用域之中声明函数。ES6 规定,块级作用域之中,函数声明语句的行为类似于let在块级作用域之外不可引用

技术分享图片

 

 上面代码在 ES5 中运行,会得到“I am inside!”,因为在if内声明的函数f会被提升到函数头部

 

上面代码在 ES5 中运行,会得到“I am inside!”,因为在if内声明的函数f会被提升到函数头部,实际运行的代码如下。

技术分享图片

let 和 const 命令

原文:https://www.cnblogs.com/Rivend/p/12076393.html

(0)
(0)
   
举报
评论 一句话评论(0
关于我们 - 联系我们 - 留言反馈 - 联系我们:wmxa8@hotmail.com
© 2014 bubuko.com 版权所有
打开技术之扣,分享程序人生!