JavaScript作用域和作用域鏈

前言


JavaScript中有一個(gè)被稱為作用域(Scope)的特性猴蹂。對(duì)于許多人來(lái)說(shuō)瓶摆,作用域的概念并不是很容易理解榆鼠,本文我會(huì)盡我所能用最簡(jiǎn)單的方式來(lái)解釋作用域和作用域鏈遥昧,希望大家有所收獲!

作用域(Scope)


1.什么是作用域

作用域是在運(yùn)行時(shí)代碼中的某些特定部分中變量褥琐,函數(shù)和對(duì)象的可訪問(wèn)性锌俱。換句話說(shuō),作用域決定了代碼區(qū)塊中變量和其他資源的可見(jiàn)性敌呈∶澈辏可能這兩句話并不好理解,我們先來(lái)看個(gè)例子:

function outFun2() {
    var inVariable = "內(nèi)層變量2";
}
outFun2();//要先執(zhí)行這個(gè)函數(shù)驱富,否則根本不知道里面是啥
console.log(inVariable); // Uncaught ReferenceError: inVariable is not defined

從上面的例子可以體會(huì)到作用域的概念锚赤,變量inVariable在全局作用域沒(méi)有聲明,所以在全局作用域下取值會(huì)報(bào)錯(cuò)褐鸥。我們可以這樣理解:作用域就是一個(gè)獨(dú)立的地盤(pán)线脚,讓變量不會(huì)外泄、暴露出去叫榕。也就是說(shuō)作用域最大的用處就是隔離變量浑侥,不同作用域下同名變量不會(huì)有沖突。

ES6 之前 JavaScript 沒(méi)有塊級(jí)作用域,只有全局作用域和函數(shù)作用域晰绎。 ES6的到來(lái)寓落,為我們提供了塊級(jí)作用域,可通過(guò)新增命令let和const來(lái)體現(xiàn)。

2.全局作用域和函數(shù)作用域

在代碼中任何地方都能訪問(wèn)到的對(duì)象擁有全局作用域荞下,一般來(lái)說(shuō)以下幾種情形擁有全局作用域:

  • 最外層函數(shù) 和 在最外層函數(shù)外面定義的變量擁有全局作用域
var outVariable = "我是最外層變量"; //最外層變量
function outFun() { //最外層函數(shù)
    var inVariable = "內(nèi)層變量";
    function innerFun() { //內(nèi)層函數(shù)
        console.log(inVariable);
    }
    innerFun();
}
console.log(outVariable); //我是最外層變量
outFun(); //內(nèi)層變量
console.log(inVariable); //inVariable is not defined
innerFun(); //innerFun is not defined
  • 所有末定義直接賦值的變量自動(dòng)聲明為擁有全局作用域
function outFun2() {
    variable = "未定義直接賦值的變量";
    var inVariable2 = "內(nèi)層變量2";
}
outFun2();//要先執(zhí)行這個(gè)函數(shù)伶选,否則根本不知道里面是啥
console.log(variable); //未定義直接賦值的變量
console.log(inVariable2); //inVariable2 is not defined
  • 所有window對(duì)象的屬性擁有全局作用域

一般情況下,window對(duì)象的內(nèi)置屬性都擁有全局作用域尖昏,例如window.name仰税、window.location、window.top等等抽诉。

全局作用域有個(gè)弊端:如果我們寫(xiě)了很多行 JS 代碼陨簇,變量定義都沒(méi)有用函數(shù)包括,那么它們就全部都在全局作用域中迹淌。這樣就會(huì) 污染全局命名空間, 容易引起命名沖突河绽。

// 張三寫(xiě)的代碼中
var data = {a: 100}

// 李四寫(xiě)的代碼中
var data = {x: true}

這就是為何 jQuery、Zepto 等庫(kù)的源碼唉窃,所有的代碼都會(huì)放在(function(){....})()中耙饰。因?yàn)榉旁诶锩娴乃凶兞浚疾粫?huì)被外泄和暴露纹份,不會(huì)污染到外面榔幸,不會(huì)對(duì)其他的庫(kù)或者 JS 腳本造成影響。這是函數(shù)作用域的一個(gè)體現(xiàn)矮嫉。

函數(shù)作用域,是指聲明在函數(shù)內(nèi)部的變量削咆,和全局作用域相反,局部作用域一般只在固定的代碼片段內(nèi)可訪問(wèn)到蠢笋,最常見(jiàn)的例如函數(shù)內(nèi)部拨齐。

function doSomething(){
    var blogName="浪里行舟";
    function innerSay(){
        alert(blogName);
    }
    innerSay();
}
alert(blogName); //腳本錯(cuò)誤
innerSay(); //腳本錯(cuò)誤

作用域是分層的,內(nèi)層作用域可以訪問(wèn)外層作用域的變量昨寞,反之則不行瞻惋。 我們看個(gè)例子,用泡泡來(lái)比喻作用域可能好理解一點(diǎn):

作用域.png

最后輸出的結(jié)果為 2, 4, 12

  • 泡泡1是全局作用域援岩,有標(biāo)識(shí)符foo歼狼;

  • 泡泡2是作用域foo,有標(biāo)識(shí)符a,bar,b享怀;

  • 泡泡3是作用域bar羽峰,僅有標(biāo)識(shí)符c。

值得注意的是:塊語(yǔ)句(大括號(hào)“{}”中間的語(yǔ)句)添瓷,如 if 和 switch 條件語(yǔ)句或 for 和 while 循環(huán)語(yǔ)句梅屉,不像函數(shù),它們不會(huì)創(chuàng)建一個(gè)新的作用域鳞贷。 在塊語(yǔ)句中定義的變量將保留在它們已經(jīng)存在的作用域中坯汤。

if (true) {
    // 'if' 條件語(yǔ)句塊不會(huì)創(chuàng)建一個(gè)新的作用域
    var name = 'Hammad'; // name 依然在全局作用域中
}
console.log(name); // logs 'Hammad'

JS的初學(xué)者經(jīng)常需要花點(diǎn)時(shí)間才能習(xí)慣變量提升,而如果不理解這種特有行為搀愧,就可能導(dǎo)致 bug 惰聂。正因?yàn)槿绱耍?ES6 引入了塊級(jí)作用域,讓變量的生命周期更加可控咱筛。

塊級(jí)作用域

塊級(jí)作用域可通過(guò)新增命令let和const聲明搓幌,所聲明的變量在指定塊的作用域外無(wú)法被訪問(wèn)。塊級(jí)作用域在如下情況被創(chuàng)建:

  1. 在一個(gè)函數(shù)內(nèi)部
  2. 在一個(gè)代碼塊(由一對(duì)花括號(hào)包裹)內(nèi)部

let 聲明的語(yǔ)法與 var 的語(yǔ)法一致眷蚓。你基本上可以用 let 來(lái)代替 var 進(jìn)行變量聲明鼻种,但會(huì)將變量的作用域限制在當(dāng)前代碼塊中。塊級(jí)作用域有以下幾個(gè)特點(diǎn):

  • 聲明變量不會(huì)提升到代碼塊頂部

let/const 聲明并不會(huì)被提升到當(dāng)前代碼塊的頂部沙热,因此你需要手動(dòng)將 let/const 聲明放置到頂部叉钥,以便讓變量在整個(gè)代碼塊內(nèi)部可用。

function getValue(condition) {
if (condition) {
let value = "blue";
return value;
} else {
// value 在此處不可用
return null;
}
// value 在此處不可用
}
  • 禁止重復(fù)聲明

如果一個(gè)標(biāo)識(shí)符已經(jīng)在代碼塊內(nèi)部被定義篙贸,那么在此代碼塊內(nèi)使用同一個(gè)標(biāo)識(shí)符進(jìn)行 let 聲明就會(huì)導(dǎo)致拋出錯(cuò)誤投队。例如:

var count = 30;
let count = 40; // Uncaught SyntaxError: Identifier 'count' has already been declared

在本例中, count 變量被聲明了兩次:一次使用 var 爵川,另一次使用 let 敷鸦。因?yàn)?let 不能在同一作用域內(nèi)重復(fù)聲明一個(gè)已有標(biāo)識(shí)符,此處的 let 聲明就會(huì)拋出錯(cuò)誤。但如果在嵌套的作用域內(nèi)使用 let 聲明一個(gè)同名的新變量扒披,則不會(huì)拋出錯(cuò)誤值依。

var count = 30;
// 不會(huì)拋出錯(cuò)誤
if (condition) {
let count = 40;
// 其他代碼
}
  • 循環(huán)中的綁定塊作用域的妙用

開(kāi)發(fā)者可能最希望實(shí)現(xiàn)for循環(huán)的塊級(jí)作用域了,因?yàn)榭梢园崖暶鞯挠?jì)數(shù)器變量限制在循環(huán)內(nèi)碟案,例如:

for (let i = 0; i < 10; i++) {
  // ...
}
console.log(i);
// ReferenceError: i is not defined

上面代碼中愿险,計(jì)數(shù)器i只在for循環(huán)體內(nèi)有效,在循環(huán)體外引用就會(huì)報(bào)錯(cuò)价说。

var a = [];
for (var i = 0; i < 10; i++) {
  a[i] = function () {
    console.log(i);
  };
}
a[6](); // 10

上面代碼中辆亏,變量i是var命令聲明的,在全局范圍內(nèi)都有效鳖目,所以全局只有一個(gè)變量i扮叨。每一次循環(huán),變量i的值都會(huì)發(fā)生改變领迈,而循環(huán)內(nèi)被賦給數(shù)組a的函數(shù)內(nèi)部的console.log(i)彻磁,里面的i指向的就是全局的i。也就是說(shuō)惦费,所有數(shù)組a的成員里面的i兵迅,指向的都是同一個(gè)i,導(dǎo)致運(yùn)行時(shí)輸出的是最后一輪的i的值薪贫,也就是 10恍箭。
如果使用let,聲明的變量?jī)H在塊級(jí)作用域內(nèi)有效瞧省,最后輸出的是 6扯夭。

var a = [];
for (let i = 0; i < 10; i++) {
  a[i] = function () {
    console.log(i);
  };
}
a[6](); // 6

上面代碼中,變量i是let聲明的鞍匾,當(dāng)前的i只在本輪循環(huán)有效交洗,所以每一次循環(huán)的i其實(shí)都是一個(gè)新的變量,所以最后輸出的是6橡淑。你可能會(huì)問(wèn)构拳,如果每一輪循環(huán)的變量i都是重新聲明的,那它怎么知道上一輪循環(huán)的值梁棠,從而計(jì)算出本輪循環(huán)的值置森?這是因?yàn)?JavaScript 引擎內(nèi)部會(huì)記住上一輪循環(huán)的值,初始化本輪的變量i時(shí)符糊,就在上一輪循環(huán)的基礎(chǔ)上進(jìn)行計(jì)算凫海。
另外,for循環(huán)還有一個(gè)特別之處男娄,就是設(shè)置循環(huán)變量的那部分是一個(gè)父作用域行贪,而循環(huán)體內(nèi)部是一個(gè)單獨(dú)的子作用域漾稀。

for (let i = 0; i < 3; i++) {
  let i = 'abc';
  console.log(i);
}
// abc
// abc
// abc

上面代碼正確運(yùn)行,輸出了 3 次abc建瘫。這表明函數(shù)內(nèi)部的變量i與循環(huán)變量i不在同一個(gè)作用域崭捍,有各自單獨(dú)的作用域。

作用域鏈


1.什么是自由變量

首先認(rèn)識(shí)一下什么叫做 自由變量 暖混。如下代碼中缕贡,console.log(a)要得到a變量,但是在當(dāng)前的作用域中沒(méi)有定義a(可對(duì)比一下b)拣播。當(dāng)前作用域沒(méi)有定義的變量,這成為 自由變量 收擦。自由變量的值如何得到 —— 向父級(jí)作用域?qū)ふ遥ㄗ⒁猓哼@種說(shuō)法并不嚴(yán)謹(jǐn)贮配,下文會(huì)重點(diǎn)解釋)。

var a = 100
function fn() {
    var b = 200
    console.log(a) // 這里的a在這里就是一個(gè)自由變量
    console.log(b)
}
fn()

2.什么是作用域鏈

如果父級(jí)也沒(méi)呢塞赂?再一層一層向上尋找泪勒,直到找到全局作用域還是沒(méi)找到,就宣布放棄宴猾。這種一層一層的關(guān)系圆存,就是 作用域鏈 。

var a = 100
function F1() {
    var b = 200
    function F2() {
        var c = 300
        console.log(a) // 自由變量仇哆,順作用域鏈向父作用域找
        console.log(b) // 自由變量沦辙,順作用域鏈向父作用域找
        console.log(c) // 本作用域的變量
    }
    F2()
}
F1()

3.關(guān)于自由變量的取值

關(guān)于自由變量的值,上文提到要到父作用域中取讹剔,其實(shí)有時(shí)候這種解釋會(huì)產(chǎn)生歧義油讯。

var x = 10
function fn() {
  console.log(x)
}
function show(f) {
  var x = 20
  (function() {
    f() //10,而不是20
  })()
}
show(fn)

在fn函數(shù)中延欠,取自由變量x的值時(shí)陌兑,要到哪個(gè)作用域中取由捎?——要到創(chuàng)建fn函數(shù)的那個(gè)作用域中取兔综,無(wú)論fn函數(shù)將在哪里調(diào)用
所以狞玛,不要在用以上說(shuō)法了软驰。相比而言,用這句話描述會(huì)更加貼切:要到創(chuàng)建這個(gè)函數(shù)的那個(gè)域”为居。
作用域中取值,這里強(qiáng)調(diào)的是“創(chuàng)建”碌宴,而不是“調(diào)用”
,切記切記——其實(shí)這就是所謂的"靜態(tài)作用域"

var a = 10
function fn() {
  var b = 20
  function bar() {
    console.log(a + b) //30
  }
  return bar
}
var x = fn(),
  b = 200
x() //bar()

fn()返回的是bar函數(shù)蒙畴,賦值給x贰镣。執(zhí)行x()呜象,即執(zhí)行bar函數(shù)代碼。取b的值時(shí)碑隆,直接在fn作用域取出恭陡。取a的值時(shí),試圖在fn作用域取上煤,但是取不到休玩,只能轉(zhuǎn)向創(chuàng)建fn的那個(gè)作用域中去查找,結(jié)果找到了,所以最后的結(jié)果是30

作用域與執(zhí)行上下文


許多開(kāi)發(fā)人員經(jīng)辰俸荩混淆作用域和執(zhí)行上下文的概念拴疤,誤認(rèn)為它們是相同的概念,但事實(shí)并非如此独泞。

我們知道JavaScript屬于解釋型語(yǔ)言呐矾,JavaScript的執(zhí)行分為:解釋和執(zhí)行兩個(gè)階段,這兩個(gè)階段所做的事并不一樣:

解釋階段:

  • 詞法分析

  • 語(yǔ)法分析

  • 作用域規(guī)則確定

執(zhí)行階段:

  • 創(chuàng)建執(zhí)行上下文

  • 執(zhí)行函數(shù)代碼

  • 垃圾回收

JavaScript解釋階段便會(huì)確定作用域規(guī)則,因此作用域在函數(shù)定義時(shí)就已經(jīng)確定了懦砂,而不是在函數(shù)調(diào)用時(shí)確定蜒犯,但是執(zhí)行上下文是函數(shù)執(zhí)行之前創(chuàng)建的。執(zhí)行上下文最明顯的就是this的指向是執(zhí)行時(shí)確定的荞膘。而作用域訪問(wèn)的變量是編寫(xiě)代碼的結(jié)構(gòu)確定的罚随。

作用域和執(zhí)行上下文之間最大的區(qū)別是:
執(zhí)行上下文在運(yùn)行時(shí)確定,隨時(shí)可能改變羽资;作用域在定義時(shí)就確定淘菩,并且不會(huì)改變

一個(gè)作用域下可能包含若干個(gè)上下文環(huán)境削罩。有可能從來(lái)沒(méi)有過(guò)上下文環(huán)境(函數(shù)從來(lái)就沒(méi)有被調(diào)用過(guò))瞄勾;有可能有過(guò)弥激,現(xiàn)在函數(shù)被調(diào)用完畢后,上下文環(huán)境被銷毀了微服;有可能同時(shí)存在一個(gè)或多個(gè)(閉包)。同一個(gè)作用域下糙麦,不同的調(diào)用會(huì)產(chǎn)生不同的執(zhí)行上下文環(huán)境,繼而產(chǎn)生不同的變量的值 赡磅。

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末宝与,一起剝皮案震驚了整個(gè)濱河市冶匹,隨后出現(xiàn)的幾起案子咆瘟,更是在濱河造成了極大的恐慌,老刑警劉巖袒餐,帶你破解...
    沈念sama閱讀 216,843評(píng)論 6 502
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異卧檐,居然都是意外死亡幢炸,警方通過(guò)查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,538評(píng)論 3 392
  • 文/潘曉璐 我一進(jìn)店門(mén),熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)逻澳,“玉大人,你說(shuō)我怎么就攤上這事斜做。” “怎么了瓤逼?”我有些...
    開(kāi)封第一講書(shū)人閱讀 163,187評(píng)論 0 353
  • 文/不壞的土叔 我叫張陵霸旗,是天一觀的道長(zhǎng)。 經(jīng)常有香客問(wèn)我诱告,道長(zhǎng),這世上最難降的妖魔是什么精居? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 58,264評(píng)論 1 292
  • 正文 為了忘掉前任靴姿,我火速辦了婚禮沃但,結(jié)果婚禮上佛吓,老公的妹妹穿的比我還像新娘垂攘。我一直安慰自己坝疼,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,289評(píng)論 6 390
  • 文/花漫 我一把揭開(kāi)白布仪芒。 她就那樣靜靜地躺著,像睡著了一般掂名。 火紅的嫁衣襯著肌膚如雪哟沫。 梳的紋絲不亂的頭發(fā)上,一...
    開(kāi)封第一講書(shū)人閱讀 51,231評(píng)論 1 299
  • 那天嗜诀,我揣著相機(jī)與錄音,去河邊找鬼发皿。 笑死拂蝎,一個(gè)胖子當(dāng)著我的面吹牛穴墅,可吹牛的內(nèi)容都是我干的温自。 我是一名探鬼主播,決...
    沈念sama閱讀 40,116評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼松捉,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼券躁!你這毒婦竟也來(lái)了?” 一聲冷哼從身側(cè)響起也拜,我...
    開(kāi)封第一講書(shū)人閱讀 38,945評(píng)論 0 275
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎蔓钟,沒(méi)想到半個(gè)月后卵贱,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體滥沫,經(jīng)...
    沈念sama閱讀 45,367評(píng)論 1 313
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡兰绣,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,581評(píng)論 2 333
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了缀辩。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 39,754評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡瓢阴,死狀恐怖健无,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情累贤,我是刑警寧澤,帶...
    沈念sama閱讀 35,458評(píng)論 5 344
  • 正文 年R本政府宣布痹束,位于F島的核電站,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏屎媳。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,068評(píng)論 3 327
  • 文/蒙蒙 一风响、第九天 我趴在偏房一處隱蔽的房頂上張望丹禀。 院中可真熱鬧,春花似錦双泪、人聲如沸。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 31,692評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)贫导。三九已至,卻和暖如春孩灯,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背峰档。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 32,842評(píng)論 1 269
  • 我被黑心中介騙來(lái)泰國(guó)打工面哥, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人尚卫。 一個(gè)月前我還...
    沈念sama閱讀 47,797評(píng)論 2 369
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像刹泄,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子特石,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,654評(píng)論 2 354

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

  • 前言 JavaScript中有一個(gè)被稱為作用域(Scope)的特性姆蘸。雖然對(duì)于許多新手開(kāi)發(fā)者來(lái)說(shuō),作用域的概念并不是...
    前端三少爺閱讀 172評(píng)論 0 0
  • 作用域(Scope) 什么是作用域 作用域是在運(yùn)行時(shí)代碼中的某些特定部分中變量逞敷,函數(shù)和對(duì)象的可訪問(wèn)性灌侣。換句話說(shuō),作...
    oWSQo閱讀 190評(píng)論 0 0
  • 作者:浪里行舟 Fundebug經(jīng)授權(quán)轉(zhuǎn)載牛柒,版權(quán)歸原作者所有。 前言 JavaScript 中有一個(gè)被稱為作用域(...
    Fundebug閱讀 7,873評(píng)論 0 15
  • 前言 JavaScript 中有一個(gè)被稱為作用域(Scope)的特性皮壁。雖然對(duì)于許多新手開(kāi)發(fā)者來(lái)說(shuō)符喝,作用域的概念并不...
    CodeMT閱讀 100評(píng)論 0 0
  • 現(xiàn)在該講作用域鏈了。在本文中畏腕,我假設(shè)你了解執(zhí)行上下文的基礎(chǔ)知識(shí):盡管如此,我也會(huì)很快就此發(fā)表一篇文章描馅。?? 我們來(lái)看...
    xiaojichao閱讀 166評(píng)論 0 0