JavaScript作用域和作用域鏈

作用域(Scope)

什么是作用域

作用域是在運行時代碼中的某些特定部分中變量和媳,函數和對象的可訪問性格遭。換句話說,作用域決定了代碼區(qū)塊中變量和其他資源的可見性留瞳。

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

從上面的例子可以體會到作用域的概念,變量inVariable在全局作用域沒有聲明她倘,所以在全局作用域下取值會報錯璧微。我們可以這樣理解:作用域就是一個獨立的地盤,讓變量不會外泄硬梁、暴露出去前硫。也就是說作用域最大的用處就是隔離變量,不同作用域下同名變量不會有沖突荧止。
ES6 之前 JavaScript 沒有塊級作用域,只有全局作用域和函數作用域屹电。ES6的到來阶剑,為我們提供了‘塊級作用域’,可通過新增命令let和const來體現。

全局作用域和函數作用域

在代碼中任何地方都能訪問到的對象擁有全局作用域危号,一般來說以下幾種情形擁有全局作用域:

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

一般情況下,window對象的內置屬性都擁有全局作用域外莲,例如window.name猪半、window.locationwindow.top等等苍狰。
全局作用域有個弊端:如果我們寫了很多行JS代碼办龄,變量定義都沒有用函數包括,那么它們就全部都在全局作用域中淋昭。這樣就會 污染全局命名空間, 容易引起命名沖突俐填。

// 張三寫的代碼中
var data = {a: 100}
// 李四寫的代碼中
var data = {x: true}

這就是為何 jQuery、Zepto等庫的源碼翔忽,所有的代碼都會放在(function(){....})()中英融。因為放在里面的所有變量,都不會被外泄和暴露歇式,不會污染到外面驶悟,不會對其他的庫或者JS腳本造成影響。這是函數作用域的一個體現材失。
函數作用域是指聲明在函數內部的變量痕鳍,和全局作用域相反,局部作用域一般只在固定的代碼片段內可訪問到龙巨,最常見的例如函數內部笼呆。

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

作用域是分層的,內層作用域可以訪問外層作用域的變量旨别,反之則不行诗赌。

塊級作用域

塊級作用域可通過letconst聲明,所聲明的變量在指定塊的作用域外無法被訪問秸弛。塊級作用域在如下情況被創(chuàng)建:

  • 在一個函數內部
  • 在一個代碼塊(由一對花括號包裹)內部

塊級作用域有以下幾個特點:

  • 聲明變量不會提升到代碼塊頂部
  • 禁止重復聲明
  • 循環(huán)中的綁定塊作用域的妙用

開發(fā)者可能最希望實現for循環(huán)的塊級作用域了铭若,因為可以把聲明的計數器變量限制在循環(huán)內,例如:

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

上面代碼中递览,計數器i只在for循環(huán)體內有效叼屠,在循環(huán)體外引用就會報錯。

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

上面代碼中绞铃,變量ivar命令聲明的环鲤,在全局范圍內都有效,所以全局只有一個變量i憎兽。每一次循環(huán)冷离,變量i的值都會發(fā)生改變,而循環(huán)內被賦給數組a的函數內部的console.log(i)纯命,里面的i指向的就是全局的i西剥。也就是說,所有數組a的成員里面的i亿汞,指向的都是同一個i瞭空,導致運行時輸出的是最后一輪的i的值,也就是 10疗我。
如果使用let咆畏,聲明的變量僅在塊級作用域內有效,最后輸出的是6吴裤。

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

上面代碼中旧找,變量ilet聲明的,當前的i只在本輪循環(huán)有效麦牺,所以每一次循環(huán)的i其實都是一個新的變量钮蛛,所以最后輸出的是6。你可能會問剖膳,如果每一輪循環(huán)的變量i都是重新聲明的魏颓,那它怎么知道上一輪循環(huán)的值,從而計算出本輪循環(huán)的值吱晒?這是因為JavaScript引擎內部會記住上一輪循環(huán)的值甸饱,初始化本輪的變量i時,就在上一輪循環(huán)的基礎上進行計算仑濒。
另外叹话,for循環(huán)還有一個特別之處,就是設置循環(huán)變量的那部分是一個父作用域躏精,而循環(huán)體內部是一個單獨的子作用域渣刷。

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

上面代碼正確運行,輸出了3次abc矗烛。這表明函數內部的變量i與循環(huán)變量i不在同一個作用域辅柴,有各自單獨的作用域。

作用域鏈

當獲取某個變量時瞭吃,如果這個變量在當前作用域內沒有定義該變量碌嘀,就向向父級作用域尋找。如果父級也沒有歪架,就一層一層向上尋找股冗,直到找到全局作用域還是沒找到,就宣布放棄和蚪。這種一層一層的關系止状,就是作用域鏈 烹棉。

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

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

  • 解釋階段:詞法分析怯疤、語法分析浆洗、作用域規(guī)則確定。
  • 執(zhí)行階段:創(chuàng)建執(zhí)行上下文集峦、執(zhí)行函數代碼伏社、垃圾回收。

JavaScript解釋階段便會確定作用域規(guī)則塔淤,因此作用域在函數定義時就已經確定了摘昌,而不是在函數調用時確定,但是執(zhí)行上下文是函數執(zhí)行之前創(chuàng)建的高蜂。執(zhí)行上下文最明顯的就是this的指向是執(zhí)行時確定的聪黎。而作用域訪問的變量是編寫代碼的結構確定的。

作用域和執(zhí)行上下文之間最大的區(qū)別是:
執(zhí)行上下文在運行時確定妨马,隨時可能改變挺举;作用域在定義時就確定,并且不會改變烘跺。

一個作用域下可能包含若干個上下文環(huán)境湘纵。有可能從來沒有過上下文環(huán)境(函數從來就沒有被調用過);有可能有過滤淳,現在函數被調用完畢后梧喷,上下文環(huán)境被銷毀了;有可能同時存在一個或多個(閉包)脖咐。同一個作用域下铺敌,不同的調用會產生不同的執(zhí)行上下文環(huán)境,繼而產生不同的變量的值屁擅。

?著作權歸作者所有,轉載或內容合作請聯系作者
  • 序言:七十年代末偿凭,一起剝皮案震驚了整個濱河市,隨后出現的幾起案子派歌,更是在濱河造成了極大的恐慌弯囊,老刑警劉巖,帶你破解...
    沈念sama閱讀 212,383評論 6 493
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件胶果,死亡現場離奇詭異匾嘱,居然都是意外死亡,警方通過查閱死者的電腦和手機早抠,發(fā)現死者居然都...
    沈念sama閱讀 90,522評論 3 385
  • 文/潘曉璐 我一進店門霎烙,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事悬垃∮沃纾” “怎么了?”我有些...
    開封第一講書人閱讀 157,852評論 0 348
  • 文/不壞的土叔 我叫張陵盗忱,是天一觀的道長酱床。 經常有香客問我,道長趟佃,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 56,621評論 1 284
  • 正文 為了忘掉前任昧捷,我火速辦了婚禮闲昭,結果婚禮上,老公的妹妹穿的比我還像新娘靡挥。我一直安慰自己序矩,他們只是感情好,可當我...
    茶點故事閱讀 65,741評論 6 386
  • 文/花漫 我一把揭開白布跋破。 她就那樣靜靜地躺著簸淀,像睡著了一般。 火紅的嫁衣襯著肌膚如雪毒返。 梳的紋絲不亂的頭發(fā)上租幕,一...
    開封第一講書人閱讀 49,929評論 1 290
  • 那天,我揣著相機與錄音拧簸,去河邊找鬼劲绪。 笑死,一個胖子當著我的面吹牛盆赤,可吹牛的內容都是我干的贾富。 我是一名探鬼主播,決...
    沈念sama閱讀 39,076評論 3 410
  • 文/蒼蘭香墨 我猛地睜開眼牺六,長吁一口氣:“原來是場噩夢啊……” “哼颤枪!你這毒婦竟也來了?” 一聲冷哼從身側響起淑际,我...
    開封第一講書人閱讀 37,803評論 0 268
  • 序言:老撾萬榮一對情侶失蹤畏纲,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后庸追,有當地人在樹林里發(fā)現了一具尸體霍骄,經...
    沈念sama閱讀 44,265評論 1 303
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 36,582評論 2 327
  • 正文 我和宋清朗相戀三年淡溯,在試婚紗的時候發(fā)現自己被綠了读整。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 38,716評論 1 341
  • 序言:一個原本活蹦亂跳的男人離奇死亡咱娶,死狀恐怖米间,靈堂內的尸體忽然破棺而出强品,到底是詐尸還是另有隱情,我是刑警寧澤屈糊,帶...
    沈念sama閱讀 34,395評論 4 333
  • 正文 年R本政府宣布的榛,位于F島的核電站,受9級特大地震影響逻锐,放射性物質發(fā)生泄漏夫晌。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 40,039評論 3 316
  • 文/蒙蒙 一昧诱、第九天 我趴在偏房一處隱蔽的房頂上張望晓淀。 院中可真熱鬧,春花似錦盏档、人聲如沸凶掰。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,798評論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽懦窘。三九已至,卻和暖如春稚配,著一層夾襖步出監(jiān)牢的瞬間畅涂,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 32,027評論 1 266
  • 我被黑心中介騙來泰國打工药有, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留毅戈,地道東北人。 一個月前我還...
    沈念sama閱讀 46,488評論 2 361
  • 正文 我出身青樓愤惰,卻偏偏與公主長得像苇经,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子宦言,可洞房花燭夜當晚...
    茶點故事閱讀 43,612評論 2 350

推薦閱讀更多精彩內容