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


很早就想系统的学习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,除非你确定你申明的是一个变量(会变的值)。
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,558评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,002评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,036评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,024评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,144评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,255评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,295评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,068评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,478评论 1 305
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,789评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,965评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,649评论 4 336
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,267评论 3 318
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,982评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,223评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,800评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,847评论 2 351

推荐阅读更多精彩内容