前言:
最近開始看阮一峰老師的《ECMAScript 6 入門》(以下簡稱原文)學(xué)習(xí)ECMAScript 6(下文簡稱ES6)的知識,整理出一些知識點(diǎn)加上我的理解來做成文章筆記。按照章節(jié)為單位一個章節(jié)一篇筆記。
文章代碼與目錄結(jié)構(gòu)和原文不同。
這一章原文鏈接 let 和 const 命令 。
let
let
是用來聲明一個變量。
不同與var
會存在變量提升(下文有介紹),let
所聲明的變量值只在let
命令所在的代碼塊內(nèi)有效。
同一個作用域(下文有介紹)不可使用 let
重復(fù)聲明同一個變量。
注意:
- 聲明變量
- 沒有變量提升
- 不可重復(fù)聲明
- 只在
let
命令所在代碼塊有效
let sample = 1;sample = 2;let sample = 2; // 將會報錯
{ let sample = 1; console.log(sample); // 正常輸出 1}console.log(sample); // 將會報錯,因?yàn)橹辉趌et命令所在代碼塊有效
const
const
是用來聲明一個只讀常量。
一旦聲明,常量的值就不能改變。如果試著改變常量的值會報錯。
并且const
在聲明的時候就必須對其賦值,只聲明不賦值,也會報錯。
同一個作用域不可使用 const
重復(fù)聲明同一個常量。const
與let
一樣,都因?yàn)樽饔糜蛟?,只能在所在代碼塊中有效。
const實(shí)際上保證的,并不是變量的值不得改動,而是變量指向的那個內(nèi)存地址所保存的數(shù)據(jù)不得改動。
注意:
- 聲明常量
- 聲明后不可以改變
- 聲明的時候必須對其賦值
- 不可重復(fù)聲明
- 在
const
命令所在代碼塊有效
const sample = 1;sample = 2; // 將會報錯,const 聲明的變量不可以重新賦值
const sample; // 直接報錯,const 聲明的時候必須對其賦值
let 與 const
引入let
后,已經(jīng)可以代替var
了,在let
與const
之中能用const
就盡量用const
。
let 與 const 不同處
let
與 const
的區(qū)別就是一個聲明變量一個聲明常量,變量可以重新賦值,常量不能重新賦值。
let sampleLet = 2;const sampleConst = 1;sampleLet = 3; // 正常sampleConst = 3; // 報錯
let 與 const 相同處
- 都只能先聲明后使用,不能變量提升。
- 都不可以在同一個作用域中重復(fù)聲明
- 都只在命令所在代碼塊有效
{sampleLet; // 報錯sampleConst; // 報錯let sampleLet = 2;const sampleConst = 1;let sampleLet = 3; // 報錯const sampleConst = 3; // 報錯}sampleLet; // 報錯sampleConst; // 報錯
變量提升(Hoisting)
在ES6之前,使用var
聲明變量時會產(chǎn)生一種叫做變量提升的特性。
無論是在代碼的哪個地方聲明的,都會提升到當(dāng)前作用域的最頂部,這種行為叫做變量提升。
為了糾正這種現(xiàn)象,let
命令改變了語法行為,它所聲明的變量一定要在聲明后使用,否則報錯
上文
let
與const
表示變量不能提升,真的是這樣嗎?
其實(shí)在 JavaScript 中,所有表示var,?let,?const,?function,?function*, class
的聲明都會被提升。let
與const
聲明變量會在環(huán)境實(shí)例化時被創(chuàng)建,但是在變量的詞法綁定之前不允許以任何方式對其進(jìn)行訪問,也就是說,當(dāng)你在聲明前調(diào)用變量將會報錯但是報錯信息不是未定義而是無法在初始化之前訪問。這里也就引出了下一個概念,叫做暫時性死區(qū)。
// var 聲明會變量提升,不會報錯,但是值為 undefinedconsole.log(sampleVar); // undefinedvar sampleVar = 1;// let 聲明不會變量提升,但是報錯不是 not definedconsole.log(sampleLet); // Cannot access sampleLet before initializationlet sampleLet = 1;// const 聲明不會變量提升,但是報錯不是 not definedconsole.log(sampleConst); // Cannot access sampleConst before initializationconst sampleConst = 1;// 直接使用沒有聲明的變量報錯為 ” is not defined “console.log(sample); //sample is not defined
暫時性死區(qū)
ES6 規(guī)定,如果代碼區(qū)塊中存在 let
和 const
命令聲明的變量,這個區(qū)塊對這些變量從一開始就形成了封閉作用域,凡是在聲明之前就使用這些變量,就會報錯。直到聲明語句完成,這些變量才能被訪問(獲取或設(shè)置),
這在語法上稱為“暫時性死區(qū)”(英temporal dead zone,簡 TDZ),即代碼塊開始到變量聲明語句完成之間的區(qū)域。
var sample = 1;if (true) { sample = 1; // 報錯 let sample;}
簡單來說,就是let
和 const
命令聲明的變量,在進(jìn)入這個聲明代碼所在的作用域時,就已經(jīng)存在,但是不可以獲取或使用,直到聲明語句完成,才可以訪問。
塊級作用域
作用域(scope,或譯有效范圍)就是變量和函數(shù)的可訪問范圍,即作用域控制著變量和函數(shù)的可見性和生命周期。
let與const 塊級作用域
作用域并不是ES6的新東西,但是在ES5只有全局作用域和函數(shù)作用域,為了解決塊級作用域,ES6可以使用**let**
與**const**
聲明一個塊級作用域的變量。?var
聲明的變量具有變量提升特性,所以沒有塊的概念,可以跨塊訪問,但不能跨函數(shù)。
外層作用域無法讀取內(nèi)層作用域的變量。
{ // 塊作用域 var sampleVar = 1; let sampleLet = 2; const sampleConst = 3; console.log(sampleVar); // 成功輸出 1 console.log(sampleLet); // 成功輸出 2 console.log(sampleConst); // 成功輸出 3}console.log(sampleVar); // 成功輸出 1console.log(sampleLet); // 報錯 not definedconsole.log(sampleConst); // 報錯 not defined
?
ES6 允許塊級作用域的任意嵌套。
同一個作用域不可使用let
或const
聲明同一個變量,內(nèi)層作用域可以定義外層作用域的同名變量。
{ { { let sample = Hello World; // 外層作用域 { let sample = sample; } // 不報錯 { console.log(sample); } // 正常輸出 ‘Hello World’ } }}
塊級作用域與函數(shù)聲明
ES5 規(guī)定,函數(shù)只能在頂層作用域和函數(shù)作用域之中聲明,不能在塊級作用域聲明。
ES6 規(guī)定,塊級作用域之中,函數(shù)聲明語句的行為類似于**let**
,在塊級作用域之外不可引用。
/* ES5,這兩種情況都是不合法的,因?yàn)檫@兩個函數(shù)聲明都是在塊作用域中聲明。 但應(yīng)為瀏覽器為了兼容以前的舊代碼,還是支持在塊級作用域之中聲明函數(shù)。所以不會報錯*/ if (true) { function sampleFn() {}}try { function sampleFn() {}} catch(e) { // ...}
/* ES6,函數(shù)聲明語句的行為類似于let,在塊級作用域之外不可引用*/ if (true) { sampleFn(); // 正常輸出,函數(shù)聲明語句的行為類似于let function sampleFn() { console.log(Hello World); }}// 但其實(shí)在塊級作用域之外也可以引用函數(shù),只不過值為undefinedif (false) { function sampleFn() { console.log(Hello World); }}console.log(sampleFn); // 正常輸出 undefinedsampleFn(); // 報錯為sampleFffffdn is not defined
為什么塊級作用域之外也可以引用函數(shù)呢?
如果改變了塊級作用域內(nèi)聲明的函數(shù)的處理規(guī)則,顯然會對老代碼產(chǎn)生很大影響。為了減輕因此產(chǎn)生的不兼容問題,ES6 在附錄 B里面規(guī)定,瀏覽器的實(shí)現(xiàn)可以不遵守上面的規(guī)定(指函數(shù)聲明語句的行為),有自己的行為方式。
- 允許在塊級作用域內(nèi)聲明函數(shù)。
- 函數(shù)聲明類似于
**var**
,即會提升到全局作用域或函數(shù)作用域的頭部。 - 同時,函數(shù)聲明還會提升到所在的塊級作用域的頭部。
注意,上面三條規(guī)則只對 ES6 的瀏覽器實(shí)現(xiàn)有效,其他環(huán)境的實(shí)現(xiàn)不用遵守,還是將塊級作用域的函數(shù)聲明當(dāng)作let
處理。
我們應(yīng)該避免在塊級作用域內(nèi)聲明函數(shù)。如果確實(shí)需要,也應(yīng)該寫成函數(shù)表達(dá)式,而不是函數(shù)聲明語句。
// 函數(shù)聲明語句,不要在塊作用域中使用,因?yàn)闀凶兞刻嵘齵 function sampleFn() { console.log("Hello World"); }}// 函數(shù)表達(dá)式,在塊作用域中,函數(shù)不會有變量提升{ const sampleFn = function () { console.log("Hello World"); }}
頂層對象
頂層對象,在瀏覽器環(huán)境指的是window
對象。
ES5 之中,頂層對象的屬性與全局變量是等價的。
ES6 為了改變這一點(diǎn),
一方面規(guī)定,為了保持兼容性,var
命令和function
命令聲明的全局變量,依舊是頂層對象的屬性;
另一方面規(guī)定,let
命令、const
命令、class
命令聲明的全局變量,不屬于頂層對象的屬性。
/* ES5 之中,頂層對象的屬性賦值與全局變量的賦值,是同一件事。*/window.sample = 1;console.log(window.sample); // 正常輸出 1sample = 2;console.log(window.sample);// 正常輸出 2/* ES6 之中,let命令、const命令、class命令聲明的全局變量,不屬于頂層對象的屬性。*/var sampleVar = 1;console.log(window.sampleVar) // 正常輸出 1let sampleLet = 1;console.log(window.sampleLet) // 正常輸出 undefinedlet sampleConst = 1;console.log(window.sampleConst) // 正常輸出 undefined
window
提供全局環(huán)境(即全局作用域)所有代碼都是在這個環(huán)境中運(yùn)行。
函數(shù)里面的this
,如果函數(shù)不是作為對象的方法運(yùn)行,而是單純作為函數(shù)運(yùn)行,this
會指向頂層對象。但是,嚴(yán)格模式下,這時this
會返回undefined
。
不管是嚴(yán)格模式,還是普通模式,new Function(return this)()
,總是會返回全局對象。
function sampleFn(){ console.log(this);}sampleFn(); // 正常輸出 輸出全局對象 windowfunction sampleFn1(){ "use strict"; console.log(this)}sampleFn1(); // 正常輸出 undefined// 開啟嚴(yán)格模式"use strict";const sample = new Function(return this)();console.log(sample); // 正常輸出 輸出全局對象 window