「JavaScript學(xué)習(xí)筆記」 JavaScript 中的變量聲明簡(jiǎn)析

update:
2017-10-23 更新了文中一些表達(dá)以及添加了JS編譯部分的理解。
2018-06-06 這篇文章能更好的理解ES6的letconst理解ES6中的暫時(shí)死區(qū)(TDZ)


先前看到方方老師最新的文章《我用了兩個(gè)月的時(shí)間才理解 let》蜕乡,以下是對(duì)該文章的學(xué)習(xí)結(jié)合自己理解總結(jié)的筆記缠沈。

由于自己對(duì)JavaScript的編譯和運(yùn)行理解的不是非常深偶洋,文中對(duì)于JavaScript引擎的編譯部分的理解可能并不準(zhǔn)確饲趋,如有問(wèn)題歡迎提出幽邓!

var let function const聲明的過(guò)程

首先炮温,需要明白,JavaScript在準(zhǔn)備階段會(huì)由JavsScript引擎進(jìn)行編譯的牵舵,編譯的過(guò)程包括進(jìn)行詞法分析(詞法作用域)及代碼生成柒啤。當(dāng)前面的準(zhǔn)備完成后通常會(huì)立即執(zhí)行代碼。
對(duì)于變量聲明來(lái)說(shuō)畸颅,一般可以簡(jiǎn)單的分為三個(gè)過(guò)程:

  1. create創(chuàng)建 (編譯)
  2. initialize初始化 (編譯)
  3. assign賦值 (執(zhí)行代碼)

需要注意担巩,初始化當(dāng)中也可以進(jìn)行賦值。

var

// 通過(guò)fn限定作用域
function fn(){
    console.log(x) // 結(jié)果為undefined
    var x = 1
    var y = 2
}
fn()

// 1. 找到fn作用域中所有var聲明的變量没炒,create變量(x和y)
// 2. initialize變量x,y為undefined
// 3. 開(kāi)始執(zhí)行代碼 x = 1 涛癌,將變量 x assign為 1
// 4. y = 2 將變量 y assign為 2

// 可以理解為下面的情況
var x
var y 
x = 1
y = 2

基本過(guò)程:
1.找到所有var,「創(chuàng)建」變量
2.「初始化」變量并賦值undefined
3.「執(zhí)行代碼」送火,根據(jù)執(zhí)行的代碼開(kāi)始為變量「賦值」拳话。
在這個(gè)過(guò)程中產(chǎn)生了一個(gè)像是聲明提升到作用域頂部的現(xiàn)象,因此會(huì)出現(xiàn)在var x = 1之前console.log(x)漾脂。

ps:從文章中學(xué)習(xí)到假颇,聲明提升其實(shí)并非是一個(gè)官方說(shuō)法。

function

// 假設(shè)全局作用域下
fn2 ()
function fn2(){
    console.log(2)
}
// 1. 找到當(dāng)前作用域中所有的function聲明的變量骨稿,為當(dāng)前環(huán)境create這些變量(聲明fn2)笨鸡。
// 2. 初始化這些變量(fn2)并賦值為 function(){ console.log(2) }
// 3. 執(zhí)行代碼fn2()

基本過(guò)程:

  1. 會(huì)先找到所有function關(guān)鍵字的函數(shù)聲明進(jìn)行「創(chuàng)建」
  2. 進(jìn)行「初始化并賦值」的操作
  3. 「執(zhí)行」代碼fn2()姜钳,因此產(chǎn)生了函數(shù)提升的過(guò)程
    需要注意的是,這里特指函數(shù)聲明形耗。

let

function fn3(){
    let x = 1
    x = 2 
}
fn3()
// 1. fn3()作用域中找到所有的let聲明的變量并創(chuàng)建它
// 2. 開(kāi)始執(zhí)行代碼  (與var 不同哥桥,這里還沒(méi)初始化)
// 3. 執(zhí)行x = 1 ,x 「初始化」為 1(這并不是一次賦值激涤,如果代碼是 let x拟糕,就將 x 初始化為 undefined)
// 4. 執(zhí)行 x = 2,為x進(jìn)行賦值

根據(jù)這個(gè)區(qū)別倦踢,let的初始化是在執(zhí)行代碼的過(guò)程中送滞,而不是先初始化:

  1. 找到所有let「創(chuàng)建」變量。
  2. 開(kāi)始「執(zhí)行」代碼辱挥,使變量根據(jù)執(zhí)行的代碼「初始化并賦值」犁嗅,此時(shí)如果沒(méi)有賦值(比如:let x)則「初始化」為undefined
  3. 繼續(xù)執(zhí)行代碼晤碘。

由于let的初始化和賦值操作是在代碼執(zhí)行階段褂微,所以對(duì)let聲明的變量的使用必須在聲明后,也就沒(méi)有出現(xiàn)聲明提升的現(xiàn)象园爷。

const

function fn4(){
    const x = 1
    x = 3 // Uncaught TypeError: Assignment to constant variable.
}
fn4()
// 1. fn4的作用域中找到所有const聲明的變量宠蚂,并創(chuàng)建它(創(chuàng)建x) 
// 2. 為變量初始化 (變量x 初始化為3)
// 3. 執(zhí)行代碼x = 3 ,報(bào)錯(cuò)

const和let 主要區(qū)別在于const聲明的變量無(wú)法賦值童社,同時(shí)const必須在執(zhí)行初始化的時(shí)候賦值求厕,否則會(huì)拋出錯(cuò)誤Uncaught SyntaxError: Missing initializer in const declaration

letvar區(qū)別:

  1. var 的創(chuàng)建與初始化都提升了
  2. let 的創(chuàng)建提升了,但是初始化沒(méi)有提升
  3. function 的創(chuàng)建叠洗、初始化甘改、賦值都被提升了

因此在let初始化之前使用其聲明的變量,就會(huì)出現(xiàn)變量沒(méi)有定義的錯(cuò)誤:

function fn5(){
    console.log(x) // Uncaught ReferenceError: x is not defined
    let x = 1 
}
fn5()

小結(jié)

對(duì)于上面的結(jié)論灭抑,我們對(duì)JavaScript變量聲明做一些總結(jié)十艾。

創(chuàng)建:在編譯階段通過(guò)詞法分析,對(duì)關(guān)鍵字(var,let,const,function)進(jìn)行變量創(chuàng)建

表現(xiàn)為變量提升到作用域頂端進(jìn)行創(chuàng)建

初始化:仍在編譯階段腾节,不同關(guān)鍵字初始化方式不同忘嫉,主要操作是為變量賦值:

var:創(chuàng)建后初始化為undefined
let:執(zhí)行代碼階段進(jìn)行初始化,如果沒(méi)有賦值則初始化為undefined
function:函數(shù)聲明表達(dá)式為var的形式提升到作用域頂端創(chuàng)建案腺,并同時(shí)初始化庆冕,賦值為一個(gè)函數(shù)對(duì)象
const:創(chuàng)建后,執(zhí)行代碼階段進(jìn)行初始化劈榨,同時(shí)必須賦值

執(zhí)行代碼階段

var:創(chuàng)建變量時(shí)已進(jìn)行了第一次初始化
let:關(guān)鍵字后為第一次執(zhí)行访递,第一次執(zhí)行時(shí)才進(jìn)行初始化,可以不賦值同辣,不賦值則為初始化為undefined
function:
const:關(guān)鍵字后第一次執(zhí)行代碼拷姿,必須賦值
  1. 所有變量聲明方式都會(huì)將變量提升到作用域頂端惭载,并進(jìn)行創(chuàng)建
  2. 初始化的表現(xiàn)使得變量聲明不同,產(chǎn)生了變量提升的現(xiàn)象

其他相關(guān)

參考文章和知乎上的知識(shí)响巢,各瀏覽器在console上表現(xiàn)并不是一致的

image.png

比如上圖中的問(wèn)題描滔,在chrome的console中,有以下的情況:

  1. 當(dāng) let聲明變量x如果初始化失敗了踪古,就會(huì)處于一種已創(chuàng)建的狀態(tài)
  2. 此時(shí)無(wú)法再次對(duì)x進(jìn)行初始化含长,即初始化機(jī)會(huì)只有一次
  3. 此時(shí)x將處于暫時(shí)死區(qū)(temporal dead zone)
  4. 于是,此狀態(tài)下的x將是不可用的
    但是對(duì)于其他瀏覽器可能報(bào)錯(cuò)信息將不一致伏穆,具體可以參照問(wèn)題中賀老的回答拘泞。

參考:
https://zhuanlan.zhihu.com/p/28140450
https://www.zhihu.com/question/62966713

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市蜈出,隨后出現(xiàn)的幾起案子田弥,更是在濱河造成了極大的恐慌,老刑警劉巖铡原,帶你破解...
    沈念sama閱讀 218,386評(píng)論 6 506
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異商叹,居然都是意外死亡燕刻,警方通過(guò)查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,142評(píng)論 3 394
  • 文/潘曉璐 我一進(jìn)店門(mén)剖笙,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)卵洗,“玉大人,你說(shuō)我怎么就攤上這事弥咪」澹” “怎么了?”我有些...
    開(kāi)封第一講書(shū)人閱讀 164,704評(píng)論 0 353
  • 文/不壞的土叔 我叫張陵聚至,是天一觀的道長(zhǎng)酷勺。 經(jīng)常有香客問(wèn)我,道長(zhǎng)扳躬,這世上最難降的妖魔是什么脆诉? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 58,702評(píng)論 1 294
  • 正文 為了忘掉前任,我火速辦了婚禮贷币,結(jié)果婚禮上击胜,老公的妹妹穿的比我還像新娘。我一直安慰自己役纹,他們只是感情好偶摔,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,716評(píng)論 6 392
  • 文/花漫 我一把揭開(kāi)白布。 她就那樣靜靜地躺著促脉,像睡著了一般辰斋。 火紅的嫁衣襯著肌膚如雪策州。 梳的紋絲不亂的頭發(fā)上,一...
    開(kāi)封第一講書(shū)人閱讀 51,573評(píng)論 1 305
  • 那天亡呵,我揣著相機(jī)與錄音抽活,去河邊找鬼。 笑死锰什,一個(gè)胖子當(dāng)著我的面吹牛下硕,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播汁胆,決...
    沈念sama閱讀 40,314評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼梭姓,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來(lái)了嫩码?” 一聲冷哼從身側(cè)響起誉尖,我...
    開(kāi)封第一講書(shū)人閱讀 39,230評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎铸题,沒(méi)想到半個(gè)月后铡恕,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,680評(píng)論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡丢间,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,873評(píng)論 3 336
  • 正文 我和宋清朗相戀三年探熔,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片烘挫。...
    茶點(diǎn)故事閱讀 39,991評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡诀艰,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出饮六,到底是詐尸還是另有隱情其垄,我是刑警寧澤,帶...
    沈念sama閱讀 35,706評(píng)論 5 346
  • 正文 年R本政府宣布卤橄,位于F島的核電站绿满,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏虽风。R本人自食惡果不足惜棒口,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,329評(píng)論 3 330
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望辜膝。 院中可真熱鬧无牵,春花似錦、人聲如沸厂抖。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 31,910評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至七蜘,卻和暖如春谭溉,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背橡卤。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 33,038評(píng)論 1 270
  • 我被黑心中介騙來(lái)泰國(guó)打工扮念, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人碧库。 一個(gè)月前我還...
    沈念sama閱讀 48,158評(píng)論 3 370
  • 正文 我出身青樓柜与,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親嵌灰。 傳聞我的和親對(duì)象是個(gè)殘疾皇子弄匕,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,941評(píng)論 2 355

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

  • 前言 把《C++ Primer》[https://book.douban.com/subject/25708312...
    尤汐Yogy閱讀 9,518評(píng)論 1 51
  • 基本內(nèi)置類型 算術(shù)類型字符整型布爾值浮點(diǎn)數(shù) 空類型(void) 算術(shù)類型 帶符號(hào)類型和無(wú)符號(hào)類型int迁匠、short...
    2625K閱讀 3,183評(píng)論 0 1
  • 周未沒(méi)安排工作上的事!早上驹溃,剛結(jié)束八段錦城丧,女兒起床主動(dòng)要求去跑步。好開(kāi)心豌鹤!這也是她的進(jìn)步芙贫!我們跑了半小時(shí),她要學(xué)...
    真心愛(ài)禮閱讀 110評(píng)論 0 1
  • 目錄 上一章:見(jiàn)面一刻 文/余晃 我在老家沒(méi)什么朋友傍药,因此不用訪友,也不走什么親戚魂仍,所以相當(dāng)無(wú)聊拐辽。飯后就獨(dú)自在屋子...
    風(fēng)味沙拉閱讀 238評(píng)論 0 0
  • 在一片茂密森林的邊緣,一位勤勞的農(nóng)夫和他善良的妻子來(lái)到這里定居擦酌,過(guò)著平淡的生活俱诸。日出而作,日落而息赊舶。 不久睁搭,他們...
    亦橈閱讀 430評(píng)論 1 2