ES6-let 與 const


1.let命令?


基本概念

let語法類似于var,不同點(diǎn)在于let定義的變量只在定義它的代碼塊中有效。

{
var a = 1;
let b = 2;
}
a // 輸出1
b // 報(bào)錯(cuò)?Uncaught ReferenceError: b is not defined

var 定義的變量要么為全局變量,要么是在函數(shù)之中的局部變量沮尿。上述代碼塊中的?a?即為全局變量轿亮,所以在代碼塊外也可調(diào)用此變量。而在代碼塊外調(diào)用變量 b 報(bào)錯(cuò)可證明?b?只在定義它的代碼塊中有效福贞。


不存在變量提升和暫時(shí)性死區(qū)

ES6 中的?let?命令是不存在變量提升”現(xiàn)象的,變量提升指的是在變量未經(jīng)定義之前便可調(diào)用停士。

console.log(a);
var a = 1;
// undefined


console.log(b);
let b = 2;
// Uncaught ReferenceError: b is not defined

上述代碼肚医,使用 var 定義的變量 a 發(fā)生變量提升绢馍,在腳本程序運(yùn)行時(shí)變量已經(jīng)存在了,只是還未定義值肠套,所以輸出 undefined舰涌。而變量 b 在未定義之前調(diào)用打印?b?的代碼會報(bào)錯(cuò),表明使用let?命令定義的變量是不存在變量提升的你稚。

b = 3;
let b = 2;
//?Uncaught ReferenceError: b is not defined

當(dāng)你輸入上述代碼卻得到報(bào)錯(cuò)的結(jié)果是不是很疑惑呀瓷耙,為什第一句代碼沒有把變量 b 定義為一個(gè)全局變量呢?

沒錯(cuò)刁赖!“罪魁禍?zhǔn)住?就是 let 命令搁痛,因?yàn)閺漠?dāng)前作用域的頭部一直到?let 命令聲明變量?b?之前,b都是不可用的宇弛,這在語法上稱為暫時(shí)性死區(qū)(temporal dead zone)鸡典。

ES6中規(guī)定暫時(shí)性死區(qū)letconst不出現(xiàn)變量提升枪芒,能夠有效地避免在聲明變量之前就使用它彻况。


重復(fù)定義檢查

相同作用域內(nèi),var 可以讓同一個(gè)變量名在同一個(gè)作用域里被定義多次舅踪,而?let?則不允許纽甘。以下是幾個(gè)例子。

{
let a = 1;
var a = 2;
}


{
let b = 2;
let b =3;
}


function test(argument){
let argument = 4;
}
test();

上述三塊代碼均會報(bào)出變量名已經(jīng)聲明的錯(cuò)誤抽碌。


let 用途

下面考慮一種需求:需要動態(tài)往HTML中一個(gè)ID為“list”的標(biāo)簽中插入十個(gè)li標(biāo)簽悍赢,并且每個(gè)li標(biāo)簽都帶有一個(gè)提示本標(biāo)簽被點(diǎn)擊的點(diǎn)擊事件。

var list = document.getElementById('list');
for( let i=1;i<=10 ;i++){
let item = document.createElement('li');
item.appendChild(document.createTextNode('Item '+i));
item.onclick = function(e){
console.log('Item '+i+' is clicked.');
};
list.appendChild(item);
}

上述代碼利用 for 循環(huán)完成了上述需求货徙。這時(shí)候你會想這個(gè)和?let?命令有什么關(guān)系左权,我換成?var 豈不是也能實(shí)現(xiàn)。下面我們來檢驗(yàn)一下?lián)Q成?var?可行嗎痴颊?

var list = document.getElementById('list');
for( var i=1;i<=10 ;i++){
let item = document.createElement('li');
item.appendChild(document.createTextNode('Item '+i));
item.onclick = function(e){
console.log('Item '+i+' is clicked.');
};
list.appendChild(item);
}

當(dāng)我們點(diǎn)擊上述代碼生成的?li?標(biāo)簽時(shí)赏迟,會發(fā)現(xiàn)無論點(diǎn)擊哪個(gè)都會打印出“Item 11 is clicked.”。下面我來解釋一下為什么會出現(xiàn)這種情況祷舀,因?yàn)樵?for?循環(huán)中的變量?i?var定義的,在全局范圍內(nèi)都有效烹笔,而每個(gè)標(biāo)簽被點(diǎn)擊所執(zhí)行的函數(shù)內(nèi)部的?i?都指的是這個(gè)全局的?i?裳扯。而使用?let?命令時(shí),循環(huán)體的每一次執(zhí)行都產(chǎn)生一個(gè)作用域谤职,每次綁定點(diǎn)擊事件時(shí)饰豺,函數(shù)都能保留當(dāng)前計(jì)數(shù)器的數(shù)值和引用。

注意:
let允蜈、const 命令定義的全局變量不屬于頂層對象的屬性冤吨。
let a = 1;
window.a // undefined


2.const命令


基本概念

const 命令用來定義常量蒿柳,一旦聲明,不可改變漩蟆。這也意味著聲明變量的同時(shí)就需要進(jìn)行初始化垒探,不可留到以后賦值。

const Max_Age;
Max_Age = 100;
// Uncaught SyntaxError: Missing initializer in const declaration

const 命令與?let?命令一樣:
1.只在聲明的塊級作用域有效怠李;
2.常量不可提升圾叼,同樣存在暫時(shí)性死區(qū);
3.不可重復(fù)聲明捺癞。


原理

變量與內(nèi)存之間的關(guān)系由三部分組成:變量名夷蚊、內(nèi)存綁定及內(nèi)存地址。const 的實(shí)現(xiàn)原理便是在常量名和內(nèi)存地址之間創(chuàng)建一個(gè)不可變的綁定髓介。在某些情況下惕鼓,并非是值不可變的。對于基本類型(數(shù)值唐础,字符串等)而言箱歧,常量指向的內(nèi)存地址便保存著實(shí)際值。而對于對象彻犁、數(shù)組等引用類型叫胁,常量指向的只是一個(gè)指針,const 只能保證這個(gè)指針是不可變的汞幢,如下:

const obj = {};
obj.item = 456;
obj.item; // 輸出456
obj = {}; // Uncaught TypeError: Assignment to constant variable.

凍結(jié)對象

上述說明當(dāng)用?const?定義對象時(shí)驼鹅,并不能保證值不可變,下面我們就介紹如何獲取值不可變的對象森篷。除了凍結(jié)對象输钩,如果對象的屬性指向的還是對象,那么這個(gè)屬性也該被凍結(jié)仲智。如下代碼便可獲取值不可變的對象买乃。

const deepFreeze = (obj) => {
Object.freeze(obj);
Object.keys(obj).forEach((key,i) => {
if(typeof obj[key] === 'object')
{ deepFreeze(obj[key]);}
});
};


3.建議

1. 一般情況下,使用?const?命令對值進(jìn)行存儲钓辆;
2. 當(dāng)一個(gè)值容器存儲的值確認(rèn)會被改變時(shí)才使用?let?進(jìn)行定義剪验;
3. 不再使用?var

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末前联,一起剝皮案震驚了整個(gè)濱河市功戚,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌似嗤,老刑警劉巖啸臀,帶你破解...
    沈念sama閱讀 212,599評論 6 492
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異烁落,居然都是意外死亡乘粒,警方通過查閱死者的電腦和手機(jī)豌注,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,629評論 3 385
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來灯萍,“玉大人轧铁,你說我怎么就攤上這事【刮龋” “怎么了属桦?”我有些...
    開封第一講書人閱讀 158,084評論 0 348
  • 文/不壞的土叔 我叫張陵,是天一觀的道長他爸。 經(jīng)常有香客問我聂宾,道長,這世上最難降的妖魔是什么诊笤? 我笑而不...
    開封第一講書人閱讀 56,708評論 1 284
  • 正文 為了忘掉前任系谐,我火速辦了婚禮,結(jié)果婚禮上讨跟,老公的妹妹穿的比我還像新娘纪他。我一直安慰自己,他們只是感情好晾匠,可當(dāng)我...
    茶點(diǎn)故事閱讀 65,813評論 6 386
  • 文/花漫 我一把揭開白布茶袒。 她就那樣靜靜地躺著,像睡著了一般凉馆。 火紅的嫁衣襯著肌膚如雪薪寓。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 50,021評論 1 291
  • 那天澜共,我揣著相機(jī)與錄音向叉,去河邊找鬼。 笑死嗦董,一個(gè)胖子當(dāng)著我的面吹牛母谎,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播京革,決...
    沈念sama閱讀 39,120評論 3 410
  • 文/蒼蘭香墨 我猛地睜開眼奇唤,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了匹摇?” 一聲冷哼從身側(cè)響起咬扇,我...
    開封第一講書人閱讀 37,866評論 0 268
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎来惧,沒想到半個(gè)月后冗栗,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體演顾,經(jīng)...
    沈念sama閱讀 44,308評論 1 303
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡供搀,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,633評論 2 327
  • 正文 我和宋清朗相戀三年隅居,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片葛虐。...
    茶點(diǎn)故事閱讀 38,768評論 1 341
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡胎源,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出屿脐,到底是詐尸還是另有隱情涕蚤,我是刑警寧澤,帶...
    沈念sama閱讀 34,461評論 4 333
  • 正文 年R本政府宣布的诵,位于F島的核電站万栅,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏西疤。R本人自食惡果不足惜烦粒,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 40,094評論 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望代赁。 院中可真熱鬧扰她,春花似錦、人聲如沸芭碍。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,850評論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至葬馋,卻和暖如春粘昨,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背狐蜕。 一陣腳步聲響...
    開封第一講書人閱讀 32,082評論 1 267
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留卸夕,地道東北人层释。 一個(gè)月前我還...
    沈念sama閱讀 46,571評論 2 362
  • 正文 我出身青樓,卻偏偏與公主長得像快集,于是被迫代替她去往敵國和親贡羔。 傳聞我的和親對象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 43,666評論 2 350

推薦閱讀更多精彩內(nèi)容

  • let和const let命令 let用法類似var个初,但是所聲明的變涼乖寒,只在let命令所在的代碼塊內(nèi)有效 for循...
    李諾哦閱讀 362評論 0 1
  • let 和 const 命令 let 命令 塊級作用域 const 命令 頂層對象的屬性 gl...
    安小明閱讀 980評論 0 0
  • let 命令 塊級作用域 const 命令 頂層對象的屬性 global 對象 let 命令 基本用法 ES6 新...
    嘉奇呦_nice閱讀 1,624評論 0 2
  • let 命令 塊級作用域 const 命令 頂層對象的屬性 global 對象 let 命令 基本用法 ES6 新...
    卞卞村長L閱讀 591評論 0 0
  • var缺陷 1.重復(fù)聲明 2.沒有塊級作用域,只有函數(shù)作用域 let特性 1.不能重復(fù)聲明 2.作用域?yàn)閴K級 co...
    余生筑閱讀 324評論 0 0