圖解JavaScript作用域與作用域鏈

現(xiàn)在該講作用域鏈了。在本文中乘盼,我假設(shè)你了解執(zhí)行上下文的基礎(chǔ)知識(shí):盡管如此兔跌,我也會(huì)很快就此發(fā)表一篇文章。??


我們來(lái)看看如下代碼:

const name = "Lydia"
const age = 21
const city = "San Francisco"

function getPersonInfo() {
  const name = "Sarah"
  const age = 22

  return `${name} is ${age} and lives in ${city}`
}

console.log(getPersonInfo())

我們?cè)谡{(diào)用getPersonInfo()函數(shù)婉商,該函數(shù)返回一個(gè)字符串,其中包含name渣叛、agecity變量的值:Sarah is 22 and lives in San Francisco丈秩。不過(guò),getPersonInfo()函數(shù)并沒(méi)有包含名為city的變量淳衙,它是如何知道city的值的呢蘑秽?

首先饺著,內(nèi)存空間是為不同的上下文設(shè)置的。我們有默認(rèn)的全局上下文(global context)(在瀏覽器中是window肠牲,在Node中是global)幼衰,以及針對(duì)已被調(diào)用的getPersonInfo()函數(shù)的本地上下文(local context)。每個(gè)上下文還有一個(gè)作用域鏈(scope chain)缀雳。

對(duì)于getPersonInfo()函數(shù)渡嚣,作用域鏈看起來(lái)像這樣(不要擔(dān)心,現(xiàn)在還不需要完全搞清楚):

作用域鏈基本上是對(duì)對(duì)象的“引用鏈”肥印,這些對(duì)象包含對(duì)在該執(zhí)行上下文中可引用的值(和其他作用域)的引用识椰。(?:“嘿,這些都是你可以在此執(zhí)行上下文中引用的所有值”)深碱。作用域鏈?zhǔn)窃趧?chuàng)建執(zhí)行上下文時(shí)創(chuàng)建的腹鹉,這意味著它是在運(yùn)行時(shí)創(chuàng)建的!

但是莹痢,在本文中种蘸,我一般不會(huì)討論活動(dòng)對(duì)象(Activation Object)或執(zhí)行上下文(execution context),我們只關(guān)注作用域竞膳!在如下的示例中航瞭,執(zhí)行上下文中的鍵/值對(duì)表示作用域鏈中含有的對(duì)變量的引用。

全局執(zhí)行上下文的作用域鏈有對(duì)3個(gè)變量的引用:值為Lydianame坦辟,值為21age刊侯,以及值為San Franciscocity。在本地執(zhí)行上下文中锉走,有對(duì)2個(gè)變量的引用:值為Sarahname滨彻,以及值為22age

當(dāng)我們?cè)噲D訪問(wèn)getPersonInfo()函數(shù)中的變量時(shí)挪蹭,引擎會(huì)首先檢查本地作用域鏈亭饵。

本地作用域鏈中有對(duì)nameage的引用!name的值為Sarah梁厉,age的值為22辜羊。但是現(xiàn)在,試圖訪問(wèn)city時(shí)候會(huì)發(fā)生什么词顾?

為了找到city的值八秃,引擎會(huì)沿著作用域鏈向下找。這基本上只是意味著引擎不會(huì)輕易放棄:它會(huì)努力為你看看能否在本地作用域引用的外層作用域中找到變量city的值肉盹,在本例中昔驱,外層作用域是global對(duì)象

在全局執(zhí)行上下文中上忍,我們聲明了變量city骤肛,其值為San Francisco纳本,因此全局執(zhí)行上下文中有一個(gè)對(duì)變量city的引用。現(xiàn)在我們有了該變量的值腋颠,函數(shù)getPersonInfo()就可以返回字符串Sarah is 22 and lives in San Francisco ??饮醇。

我們可以沿著作用域鏈向找,但是不能沿著作用域鏈向找秕豫。好吧,這可能會(huì)令人困惑观蓄,因?yàn)橛腥苏f(shuō)的是向而不是向混移,所以我要重新表述一下:向外層作用域方向找,而不是向更內(nèi)層作用域方向找侮穿。我喜歡將這用圖形表示為一種瀑布:

甚至更深:

下面我們以這段代碼為示例歌径。

代碼幾乎是一樣的,不過(guò)有一個(gè)很大的不同點(diǎn):現(xiàn)在我們只在getPersonInfo()函數(shù)中聲明了city變量亲茅,但在全局作用域中沒(méi)有聲明回铛。我們沒(méi)有調(diào)用getPersonInfo()函數(shù),因此也沒(méi)有創(chuàng)建本地執(zhí)行上下文克锣。但是茵肃,我們?cè)噲D在全局執(zhí)行上下文中訪問(wèn)nameagecity的值袭祟。

然后它就拋出了一個(gè)ReferenceError錯(cuò)誤验残!在全局作用域中找不到一個(gè)對(duì)變量city的引用,也沒(méi)有可以查找的外層作用域巾乳,并且它不能沿著作用域向查找您没。

這樣,我們就可以把作用域作為保護(hù)變量并重用變量名的一種方法胆绊。

除了全局和本地作用域氨鹏,還有一個(gè)塊作用域。用let或者const關(guān)鍵字聲明的變量的作用域?yàn)樽罱咏拇罄ㄌ?hào)({``})压状。

const age = 21

function checkAge() {
  if (age < 21) {
    const message = "You cannot drink!"
    return message
  } else {
    const message = "You can drink!"
    return message
  }
}

可以把作用域用圖形表示為:

這里我們有一個(gè)全局作用域仆抵,一個(gè)函數(shù)作用域和兩個(gè)塊作用域。我們能兩次聲明變量message何缓,因?yàn)樵撟兞康淖饔糜蚍秶谴罄ㄌ?hào)內(nèi)肢础。

下面快速回顧一下:

  • 可以把作用域鏈看作是對(duì)我們可以在當(dāng)前執(zhí)行上下文中可以訪問(wèn)的值的一個(gè)引用鏈。
  • 作用域還讓重用在作用域鏈向下更深層次定義的變量名成為可能碌廓,因?yàn)樽兞棵荒苎刂饔糜蛳蛳抡掖洌荒芟蛏险摇?/li>

關(guān)于作用域和作用域鏈就寫(xiě)到這里了!

原文 by Lydia Hallie:https://dev.to/lydiahallie/javascript-visualized-scope-chain-13pd

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末谷婆,一起剝皮案震驚了整個(gè)濱河市慨蛙,隨后出現(xiàn)的幾起案子辽聊,更是在濱河造成了極大的恐慌,老刑警劉巖期贫,帶你破解...
    沈念sama閱讀 216,651評(píng)論 6 501
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件跟匆,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡通砍,警方通過(guò)查閱死者的電腦和手機(jī)玛臂,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,468評(píng)論 3 392
  • 文/潘曉璐 我一進(jìn)店門(mén),熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)封孙,“玉大人迹冤,你說(shuō)我怎么就攤上這事』⒓桑” “怎么了泡徙?”我有些...
    開(kāi)封第一講書(shū)人閱讀 162,931評(píng)論 0 353
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)膜蠢。 經(jīng)常有香客問(wèn)我堪藐,道長(zhǎng),這世上最難降的妖魔是什么挑围? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 58,218評(píng)論 1 292
  • 正文 為了忘掉前任礁竞,我火速辦了婚禮,結(jié)果婚禮上杉辙,老公的妹妹穿的比我還像新娘苏章。我一直安慰自己,他們只是感情好奏瞬,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,234評(píng)論 6 388
  • 文/花漫 我一把揭開(kāi)白布枫绅。 她就那樣靜靜地躺著,像睡著了一般硼端。 火紅的嫁衣襯著肌膚如雪并淋。 梳的紋絲不亂的頭發(fā)上,一...
    開(kāi)封第一講書(shū)人閱讀 51,198評(píng)論 1 299
  • 那天珍昨,我揣著相機(jī)與錄音县耽,去河邊找鬼。 笑死镣典,一個(gè)胖子當(dāng)著我的面吹牛兔毙,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播兄春,決...
    沈念sama閱讀 40,084評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼澎剥,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來(lái)了赶舆?” 一聲冷哼從身側(cè)響起哑姚,我...
    開(kāi)封第一講書(shū)人閱讀 38,926評(píng)論 0 274
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤祭饭,失蹤者是張志新(化名)和其女友劉穎,沒(méi)想到半個(gè)月后叙量,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體倡蝙,經(jīng)...
    沈念sama閱讀 45,341評(píng)論 1 311
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,563評(píng)論 2 333
  • 正文 我和宋清朗相戀三年绞佩,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了寺鸥。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 39,731評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡品山,死狀恐怖析既,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情谆奥,我是刑警寧澤,帶...
    沈念sama閱讀 35,430評(píng)論 5 343
  • 正文 年R本政府宣布拂玻,位于F島的核電站酸些,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏檐蚜。R本人自食惡果不足惜魄懂,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,036評(píng)論 3 326
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望闯第。 院中可真熱鬧市栗,春花似錦、人聲如沸咳短。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 31,676評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)咙好。三九已至篡腌,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間勾效,已是汗流浹背嘹悼。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 32,829評(píng)論 1 269
  • 我被黑心中介騙來(lái)泰國(guó)打工, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留层宫,地道東北人杨伙。 一個(gè)月前我還...
    沈念sama閱讀 47,743評(píng)論 2 368
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像萌腿,于是被迫代替她去往敵國(guó)和親限匣。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,629評(píng)論 2 354