博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
一文拨开var、let、const的云雾
阅读量:3958 次
发布时间:2019-05-24

本文共 2537 字,大约阅读时间需要 8 分钟。

var 变量提升机制

我们在全局作用域中或还是在局部作用域中,使用var关键字声明的变量,都会被提升到该作用域的最顶部,这就是我们常说的变量提升。

function person(status) {
if (status) {
var value = "小二" } else {
console.log(value) // undefined } console.log(value) // undefined}person(false)

上面example中,if代码块中的var声明的变量就被提升到了函数的顶端,有的小伙伴就会疑惑了,if代码块里的都没执行,怎么会提升到顶端了呢?,这是因为javaScript引擎,在代码预编译时,javaScript引擎会自动将所有代码里面的var关键字声明的语句都会提升到当前作用域的顶端, 因此上面的代码就会被解析为下面。

function person(status) {
var value; if (status) {
value = "小二" } else {
console.log(value) // undefined } console.log(value) // undefined}person(false)

由于javaScript存在变量提升,这让很多开发者初学起来这门语言,还得花不少时间研究变量提升,也有时在工作中因为一个变量提升导致出bug。因此Escript6中为我们带了块级声明,那么什么是块级声明呢?

  • 只在当前函数下声明的变量有效
  • 在代码块和{ }括号之内有效

let声明

let声明和var声明用法是一样,都是定义变量,使用let声明的变量没有var那样的变量提升,let声明的变量只在当前作用域中有效。我们来把上面的example重写一下。

function person(status) {
if (status) {
let value = "小二" } else {
console.log(value) // 报错 } console.log(value) // 报错}person(false)

let是块级作用域,所有外面的语句块访问不到,let是没有变量提升的,下面我们来演示一下。

console.log(value) // 报错let value = "小二"

禁止重复声明

如果在同一个作用域中某个变量已经存在,再次使用let关键字声明的话会报错。

var value = "小二"let value = "小二" // 报错// 再来看一下不同作用域的情况var value = "小二" // 全局作用域if(true) {
let value = "小二" // 代码块中声明,毫无影响}

上面example中,可以完全看到,只有在相同作用域中重复声明变量才会报错。

const声明

ECMAscript6中还提供了const关键字声明,const声明指的是常量,常量就是一旦定义完就不能修改的值。还有一点需要注意的是,常量定义必须初始化值,如果不初始化值就会报错。

const value = "小二"const age; // 报错 常量未初始化

const 与 let

const与let也没什么大不同,都是块级作用域,const常量也只会在当前代码块内有效,也不能在当前作用域中重复定义相同的变量,也不存在变量提升。

if (true) {
const name = "小二"}console.log(name) // 报错 访问不到内部变量
console.log(value) // 报错 const声明的变量也不存在变量提升const value = "小二"
let value = "小二"const value = "小二" // 报错 重复声明

const声明对象

虽然const变量不能修改指针,但是可以修改值,比如我们定义一个对象,我们就可以修改对象里的属性值,但是不可以重写整个对象。

const person = {
name: "小二", age: 23}person.age = 18 // 没问题person = {
} // 报错 不能修改对象指针

暂时死区

跟var相比,let和const定义变量不会被提升到作用域顶端,即便是用相对安全的typeof也会出现错误。

console.log(typeof value)let value = "小二"

上面example中,console.log(typeof value)会抛出错误是因为用let定义并初始化变量语句是不会执行的。此时的value还是处于在JavaScript所谓的暂时死区(temporal dead zone)简称为TDZ 中,虽然JavaScript没有明确标准TDZ,但是人们常用它描述let和const定义的变量不会提升。

我们来说一下TDZ工作原理,JavaScript引擎在扫描代码时发现变量声明时,如果遇到var就会将它们提升到当前作用域的顶端,如果遇到let或const就会将声明放到TDZ中,如果访问TDZ中的变量就会抛出错误,只有执行完TDZ中的变量才会将它移出,然后就可以正常方法。这机制只会在当前作用域生效。我们来看下不同作用域案例

console.log(typeof value)  // "undefined"if (true) {
let value = "小二"}

面说的如果变量是let和const声明的就会被放到TDZ中,前提是只会针对当前作用域内有效。所以上面代码中console.log(typeof value)不会抛出错误,let声明只会在当前的语句中有效。

觉得写得不错的话,请用你们发财的小手点个赞叭!

转载地址:http://oiozi.baihongyu.com/

你可能感兴趣的文章
软件配置管理概念-3,CM系统的概念
查看>>
JSP/Servlet应用程序优化八法
查看>>
人生必修的181条佛理
查看>>
The Most Widely Used Java Libraries
查看>>
简单在单机使用apache-james(开源邮件服务器)
查看>>
lsof 快速起步
查看>>
使用ScribeFire方便地发布blog
查看>>
跨平台Java程序注意事项
查看>>
Python字符与数字的相互转换
查看>>
C 指针解读
查看>>
有关乱码的处理---中国程序员永远无法避免的话题
查看>>
JSP的运行内幕
查看>>
python超简单的web服务器
查看>>
代理模式、静态代理、动态代理、aop
查看>>
Struts1.x Spring2.x Hibernate3.x DWR2.x整合工具文档v1.00
查看>>
大型Web2.0站点构建技术初探
查看>>
机器学习算法汇总:人工神经网络、深度学习及其它
查看>>
解决Spring中AOP不能切入Struts的DispatchAction方法的问题
查看>>
出国以后才知道英语应该怎么学
查看>>
计算机专业权威期刊投稿经验总结
查看>>