一看就懂的var、let、const三者区别

哈喽你们好,又见面啦,我依旧是那个可爱的蛙人。java

今天又周五了啊,划水人的一天,开心。安全

话很少说哈,直接搂代码函数

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关键字声明的语句都会提高到当前做用域的顶端, 所以上面的代码就会被解析为下面。code

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重写一下。ip

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中,能够彻底看到,只有在相同做用域中重复声明变量才会报错。io


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定义的变量不会提高。<br>

咱们来讲一下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声明只会在当前的语句中有效。

var let const 最大的区别

var在全局做用域声明的变量有一种行为会挂载在window对象上,它会建立一个新的全局变量做为全局对象的属性,这种行为说不定会覆盖到window对象上的某个属性,而let const声明的变量则不会有这一行为。来看下面例子。

var value1 = "张三"
let value2 = "李四"
const value3 = "王五"
console.log(window.value1) // 张三
console.log(window.value2) // undefined
console.log(window.value3) // undefined

咱们的讲解就到这里,下次见。

以为写得不错的话,请用大家发财的小手点个赞叭!

相关文章
相关标签/搜索