美文网首页让前端飞
ES6 也许你并没有完全了解let const

ES6 也许你并没有完全了解let const

作者: moonburn | 来源:发表于2017-06-22 11:39 被阅读0次

很早就想系统的学习ES6了,奈何资源太少,把阮一峰大大的ES6通读了一遍之后发现,没有一点感觉,是的,也许是自己的积累或者境界不够,真的一点感觉都没有,市面上可供学习参考的ES6的资料真的不多。之前一直盼望着《你不知道的JavaScript下卷》出中文版,听说到11月份,好吧。一次偶然,得知《高程》的作者也写了ES6的书,虽然中文版也没有,但是真的忍不了的,用我的渣英语慢慢读吧。


问在前面:

  1. 问:用const申明的对象是可以修改的,对吗?
    答:对。
  2. let申明的全局变量和用var申明的全局变量是一样的吗?
    答:不一样。

如果你都回答上了,并且知其然知其所以然,那你掌握的很好,不用往下看了。如果并不知道,那就一起慢慢回顾一下吧。


var###

相信大家已经对letconst有了一定的认识,也应该了解了块级作用域以及var并没有块级作用域。
var没有块级作用域,具体体现在两个方面,一个是在函数中,一个是在循环中。具体例子如下:

var foo = function(a){
    if(a){
        var text = 'hello'
        return text
    }else{
        //这里也有text,其值为undefined
        return null
    }
        //这里也有text,其值为undefined
}

其实我们的本意是希望如下代码:

var foo = function(a){
    if(a){
        var text = 'hello'
        return text
    }else{
        //这里text未定义
        return null
    }
        //这里text未定义
}

但是因为var没有块级作用域,并且存在变量提升,所以上述代码其实与下面的写法相同:

var foo = function(a){
     var text;undefined
    if(a){
        text = 'hello'
        return text
    }else{
        //这里也有text,其值为undefined
        return null
    }
        //这里也有text,其值为undefined
}

这是在函数中,其实在循环中也一样,比如:

for (var i = 0; i < 10; i++) {
   //做一些操作
}
//在外部,i依旧存在,并且会输出10
console.log(i); //10           

很明显,这是不符合常理的,我们期望的是:

for (var i = 0; i < 10; i++) {
   //做一些操作
}
//在外部,i未定义,报错
console.log(i); //ReferenceError       

然而并做不到。再来举个例子,在循环中的函数,使用var,是更加糟糕,更加不符合常理的,比如:

var funcs = [];
for (var i = 0; i < 10; i++) {
    funcs.push(function() { console.log(i); });
}
funcs.forEach(function(func) {
    func();     // 输出10,10次
});

其实我们期待的是,输出0-9,但是事与愿违,输出了10次10,为什么?因为for循环中的每一次迭代都共享着用var申明的i,也就是说,这里的i不是分别在单独的作用域里,而是在同一个作用域里,是一个值,而不是十个,所以,一改全改,最终变成输出10个10。
如何解决这个问题呢?
在ES5中,我们想到了用立即执行函数(IIFE),把每一个i的作用域分开。具体代码如下:

var funcs = [];
for (var i = 0; i < 10; i++) {
    funcs.push((function(value) {
        return function() {
            console.log(value);
        }
    }(i)));
}
funcs.forEach(function(func) {
    func();     //输出0-9
});

解决了,但是显得太臃肿而且难以理解是不是?是的,一眼看过去很难看清楚这个函数到底想表达什么。但是,如果我们一开始就使用let代替var,这个世界就变得简单而清爽了:

var funcs = [];
for (let i = 0; i < 10; i++) {
    funcs.push(function() { console.log(i); });
}
funcs.forEach(function(func) {
    func();     // 输出0-9
});

符合逻辑,并且代码清爽。完美!这就是let的魅力。下面让我们来深入学习let

let###

letvar差不多,只不过申明的变量有块级作用域,并且不会变量提升。什么意思呢,就是说,let声明的变量只存在于声明它的{}内部,或者循环体内部,或者函数内部,外部是不存在的。因为没有变量提升,所以有暂时性死区。

暂时性死区

这个概念我觉得是相对于var来说的,因为var有变量提升,所以先使用后申明是不会报错的,而是undefined,具体例子如下:

console.log(value);//undefined
var value = 10;
console.log(value);//10

其实这个就等于:

var value;
console.log(value);//undefined
value = 10;
console.log(value);//10

说起变量提升,在这里想多说一点。函数用function (){····}这样的方式,也有提升,而且是在变量提升之前,也就是说,提升的优先级比变量提升高,并且同名函数,后者会覆盖前者。也就是说:

foo();//b
function foo(){
    console.log('a')
}
function foo(){
    console.log('b')
}

扯远了,现在回到let的暂时性死区,和var不同,请看下面的例子:

console.log(value);//报错,ReferenceError
let value = 10;
console.log(value);

因为没有变量提升,所以只能先声明,后使用。
还要一些小细节,在同一作用域中var申明变量是可以重复申明的,并且后者覆盖前者:

var a = 10;
var a = 20;
console.log(a)//20

而在let中,这样会报错:

let a = 10;
let a = 20;//报错,语法错误,a已经申明过了。
console.log(a)

当然,这样也不行:

var a = 10;
let a = 20;//报错,语法错误,a已经申明过了。
console.log(a)

不同作用域,是可以的:

var a = 10;
function foo(){
   let a = 20;//合法
}

回到最初提出的问题之一,全局作用域下,letvar申明的变量的不同之处
首先,var大家都了解,全局申明的变量在浏览器中就是window的属性,比如:

var a = 10;
window.a//10

给个最直观的例子:

let a = 10;
window.a//报错,a未定义

明显可以看出,这是不同的。下面,我们来深入一点,用var申明全局变量并没有想象中的那么好,比如:

var RegExp = 10;
window.RegExp//10

是不是贼恐怖?把内置对象给干掉了,如此危险的操作,肯定是要避免的。用let就可以避免了:

let RegExp = 10;
window.RegExp//function RegExp() { [native code] }
console.log(RegExp )//10

为什么会这样?因为用var申明的全局变量会当做window对象的属性,而let申明的全局变量,只是一个变量,不会当做window对象的属性。let就说的差不多了,下面说下const

const###

其实constlet用法,包括需要注意的一些细节都一样比如没有变量提升,声明全局变量,块级作用域。但是还是有一些地方是不一样的,最大的一个不同点就是,const申明的变量,是不可变的,变了就报错,具体例子如下:

const a =10;
a = 20;//报错,语法错误,不能改变一个不变的值

当然,如果你以为这就是const的全部,那就不太好了,回到开头提出的问题,如果用const申明的是一个对象呢?

const a = {
    value:10
}
a.value = 20;
a;//a{value:20}

修改了,没有报错。可能你有些动摇了,那么看下一个例子:

const a = {
    value:10
}
a = {
    value:20//报错,语法错误,不能改变一个不变的值
}

结论,const只会锁定变量的地址,而不会锁定变量的属性,这个和Object.freeze(),还是有很大的区别的。

总结###

  • var没有想象中的那么好,尽量使用letconst代替。
  • var申明的全局变量,除非你真的是想给window对象加属性,不然就用letconst代替。
  • letconst没有变量提升,注意暂时性死区。
  • const并不是真正的锁定变量的所有,而只是锁定这个变量的地址而已,所以这个变量的属性是可以修改的。
  • 尽量都是用const,除非你确定你申明的是一个变量(会变的值)。

相关文章

  • ES6 也许你并没有完全了解let const

    很早就想系统的学习ES6了,奈何资源太少,把阮一峰大大的ES6通读了一遍之后发现,没有一点感觉,是的,也许是自己的...

  • (JS)

    ES6 let、const和var的区别 let和const声明变量不存在变量提升 let和const不能重复声明...

  • es6,es7,es8语法总结

    ES6 1. var let const let,const具有块级作用域,不具有变量提升 const 用...

  • ES 6 7 8 随笔 QAQ

    ES6 1. var let const let,const具有块级作用域,不具有变量提升 const 用...

  • ES 6新语法规则

    1.块级作用域(let 取代 var): ES6提出了两个新的声明变量的命令:let和const。其中,let完全...

  • ES6(let 与 const)

    ES6中引入了let 和 const来声明变量,我们来看下let 和 const 的使用 一、let let定义的...

  • react 入门基础(一)之ES6

    ####ES6 let const var 三者的区别 ### 1. let const 不能重复声明变...

  • es6总结一

    es6基础 let和const命令 let和const用于声明变量let跟var的区别 1、let定义过的变量不能...

  • ES6之 let 和 const 命令

    ES6 之 let 和 const 命令 1、let 命令 1.1、基本用法 ES6 新增了let命令,用来声明变...

  • es6解读1: let, const 和 smybol

    let const 作用域 作用域概念 如何使用let和const 使用let 形成块级作用域 es6下强制开启...

网友评论

    本文标题:ES6 也许你并没有完全了解let const

    本文链接:https://www.haomeiwen.com/subject/szvxcxtx.html