作用域(二)——詞法作用域

在上一篇文章中,我們將“作用域”定義為一套規(guī)則,這套規(guī)則用來(lái)管理引擎如何在當(dāng)前作用域以及嵌套的子作用域中根據(jù)標(biāo)識(shí)符名稱進(jìn)行變量查找奄侠。大家還可以關(guān)注我的微信公眾號(hào),蝸牛全棧载矿。

作用域共有兩種主要的工作模型垄潮。第一種是最為普通的,被大多數(shù)編程語(yǔ)言所采用的詞法作用域恢准,我們會(huì)對(duì)這種作用域進(jìn)行深入討論魂挂。另外一種叫做動(dòng)態(tài)作用域甫题,仍有一些編程語(yǔ)言在使用(比如Bash腳本馁筐、Perl中的一些模式等)。

簡(jiǎn)單的說(shuō)坠非,詞法作用域就是定義在詞法階段的作用域敏沉。換句話說(shuō),詞法作用域就是由你在寫代碼時(shí)將變量和塊作用域?qū)懺谀睦飦?lái)決定的炎码,因此當(dāng)詞法處理器處理代碼時(shí)會(huì)保持作用域不變(大部分情況是這樣的)盟迟。對(duì)一部分比較特殊的,會(huì)出現(xiàn)一些欺騙詞法作用域潦闲,主要是JavaScript中的with和eval關(guān)鍵字攒菠,非常不建議用在項(xiàng)目中。

考慮以下代碼:

function foo(a){
  var b = a * 2;
  function bar(c){
    console.log(a, b, c)
  }
  bar(b * 3);
}

foo(2); // 2, 4, 12

在這個(gè)例子中有三個(gè)逐級(jí)嵌套的作用域歉闰,為了幫助理解辖众,可以將他們想象成幾個(gè)逐級(jí)包含的氣泡卓起。

圖片

1、包含整個(gè)全局作用域凹炸,其中之后一個(gè)標(biāo)識(shí)符:foo

2戏阅、包含著foo所創(chuàng)建的作用域,其中有三個(gè)標(biāo)識(shí)符:a啤它、bar和b

3奕筐、包含著bar所創(chuàng)建的作用域,其中只有一個(gè)標(biāo)識(shí)符:c

作用域氣泡由其對(duì)應(yīng)的作用域塊代碼寫在哪決定变骡,他們是逐級(jí)包含的离赫。在后續(xù)的文章中會(huì)討論不同類型的作用域,但現(xiàn)在主要假設(shè)每一個(gè)函數(shù)都會(huì)創(chuàng)建一個(gè)新的氣泡作用域就好了塌碌。

Bar的氣泡被完全包含在foo所創(chuàng)建的氣泡中笆怠,唯一的原因是那里就是我們希望定義函數(shù)bar的位置。

注意誊爹,這里所說(shuō)的氣泡是嚴(yán)格包含的蹬刷。我們并不是在討論文氏圖這種可以跨越邊界的氣泡。換句話說(shuō)频丘,沒(méi)有任何函數(shù)的氣泡可以(部分地)同時(shí)出現(xiàn)在兩個(gè)外部作用域的氣泡中办成,就如同沒(méi)有任何函數(shù)可以部分地同時(shí)出現(xiàn)在兩個(gè)父級(jí)函數(shù)中一樣。(簡(jiǎn)單理解就是內(nèi)部作用域一定是上層作用域的子集)

作用域氣泡的結(jié)構(gòu)和互相之間的位置關(guān)系給引擎提供了足夠的位置信息搂漠,關(guān)系給引擎提供了足夠的位置信息迂卢,引擎用這些信息來(lái)查找標(biāo)識(shí)符的位置。

在上一個(gè)代碼片段中桐汤,引擎執(zhí)行console.log聲明而克,并查找a、b怔毛、c三個(gè)變量的引用员萍。它首先從最內(nèi)部的作用域,也就是bar函數(shù)的作用域氣泡開(kāi)始查找拣度。引擎無(wú)法在這里找到a碎绎,因此會(huì)去上一級(jí)到所嵌套的foo的作用域中繼續(xù)查找。在這里找到了a抗果,因此引擎使用這個(gè)引用筋帖。對(duì)b來(lái)講也是一樣的。而對(duì)c來(lái)說(shuō)冤馏,引擎在bar中就找到了它日麸。

如果a、c都存在bar和foo的內(nèi)部逮光,congsole.log就可以直接使用bar中的變量代箭,而無(wú)需到外面的foo中去查找辕录。作用域查找會(huì)在找到第一個(gè)匹配的標(biāo)識(shí)符時(shí)停止。在多層的嵌套作用域中可以定義同名的標(biāo)識(shí)符梢卸,這叫做“遮蔽效應(yīng)”(內(nèi)部的標(biāo)識(shí)符“遮蔽”了外部的標(biāo)識(shí)符)走诞。

拋開(kāi)遮蔽效應(yīng),作用域查找始終從運(yùn)行時(shí)所處的最內(nèi)部作用域開(kāi)始蛤高,逐級(jí)向外或者說(shuō)向上進(jìn)行蚣旱,直到遇見(jiàn)第一個(gè)匹配的標(biāo)識(shí)符為止。就像這樣全局變量會(huì)自動(dòng)成為全局對(duì)象(比如瀏覽器中的window對(duì)象)的屬性戴陡,因此可以不直接通過(guò)全局對(duì)象的詞法名稱塞绿,而是間接的通過(guò)全局對(duì)象屬性的引用來(lái)堆砌進(jìn)行訪問(wèn)。

window.a

通過(guò)這種技術(shù)可以訪問(wèn)那些被同名變量所遮蔽的全局變量恤批,但非全局變量如果被遮蔽了异吻,無(wú)論如何都無(wú)法被訪問(wèn)到。
就像這樣
無(wú)論函數(shù)在哪里被調(diào)用喜庞,也無(wú)論它如何被調(diào)用诀浪,他的詞法作用域都只由函數(shù)被聲明時(shí)所處的位置決定。詞法作用域查找只會(huì)查找一級(jí)標(biāo)識(shí)符延都,比如a雷猪、b和c。如果代碼中引用了foo.bar.baz晰房,詞法作用域查找只會(huì)試圖查找foo標(biāo)識(shí)符求摇,找到這個(gè)變量后,對(duì)象屬性訪問(wèn)規(guī)則會(huì)分別接管對(duì)bar和baz屬性的訪問(wèn)殊者。
參考文獻(xiàn):《你不知道的JavaScript(上)》

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末与境,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子猖吴,更是在濱河造成了極大的恐慌摔刁,老刑警劉巖,帶你破解...
    沈念sama閱讀 207,248評(píng)論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件距误,死亡現(xiàn)場(chǎng)離奇詭異簸搞,居然都是意外死亡扁位,警方通過(guò)查閱死者的電腦和手機(jī)准潭,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,681評(píng)論 2 381
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)域仇,“玉大人刑然,你說(shuō)我怎么就攤上這事∠疚瘢” “怎么了泼掠?”我有些...
    開(kāi)封第一講書人閱讀 153,443評(píng)論 0 344
  • 文/不壞的土叔 我叫張陵怔软,是天一觀的道長(zhǎng)。 經(jīng)常有香客問(wèn)我择镇,道長(zhǎng)挡逼,這世上最難降的妖魔是什么? 我笑而不...
    開(kāi)封第一講書人閱讀 55,475評(píng)論 1 279
  • 正文 為了忘掉前任腻豌,我火速辦了婚禮家坎,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘吝梅。我一直安慰自己虱疏,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 64,458評(píng)論 5 374
  • 文/花漫 我一把揭開(kāi)白布苏携。 她就那樣靜靜地躺著做瞪,像睡著了一般。 火紅的嫁衣襯著肌膚如雪右冻。 梳的紋絲不亂的頭發(fā)上装蓬,一...
    開(kāi)封第一講書人閱讀 49,185評(píng)論 1 284
  • 那天,我揣著相機(jī)與錄音纱扭,去河邊找鬼矛物。 笑死,一個(gè)胖子當(dāng)著我的面吹牛跪但,可吹牛的內(nèi)容都是我干的履羞。 我是一名探鬼主播,決...
    沈念sama閱讀 38,451評(píng)論 3 401
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼屡久,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼忆首!你這毒婦竟也來(lái)了?” 一聲冷哼從身側(cè)響起被环,我...
    開(kāi)封第一講書人閱讀 37,112評(píng)論 0 261
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤糙及,失蹤者是張志新(化名)和其女友劉穎,沒(méi)想到半個(gè)月后筛欢,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體浸锨,經(jīng)...
    沈念sama閱讀 43,609評(píng)論 1 300
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,083評(píng)論 2 325
  • 正文 我和宋清朗相戀三年版姑,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了柱搜。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 38,163評(píng)論 1 334
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡剥险,死狀恐怖聪蘸,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤健爬,帶...
    沈念sama閱讀 33,803評(píng)論 4 323
  • 正文 年R本政府宣布控乾,位于F島的核電站,受9級(jí)特大地震影響娜遵,放射性物質(zhì)發(fā)生泄漏蜕衡。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,357評(píng)論 3 307
  • 文/蒙蒙 一设拟、第九天 我趴在偏房一處隱蔽的房頂上張望衷咽。 院中可真熱鬧,春花似錦蒜绽、人聲如沸镶骗。這莊子的主人今日做“春日...
    開(kāi)封第一講書人閱讀 30,357評(píng)論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)鼎姊。三九已至,卻和暖如春相赁,著一層夾襖步出監(jiān)牢的瞬間相寇,已是汗流浹背。 一陣腳步聲響...
    開(kāi)封第一講書人閱讀 31,590評(píng)論 1 261
  • 我被黑心中介騙來(lái)泰國(guó)打工钮科, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留唤衫,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 45,636評(píng)論 2 355
  • 正文 我出身青樓绵脯,卻偏偏與公主長(zhǎng)得像佳励,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子蛆挫,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 42,925評(píng)論 2 344

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