JS進階系列之執行上下文

function test(){
     console.log(a);//undefined;
     var a = 1;
}
test();

也許你會遇到過上面這樣的面試題,你只知道它考的是變量提升,但是具體的原理又知道嗎?所以我覺得很有必要搞明白底層的原理,才能加深理解,其實圍繞的就是執行上下文的概念。

什么是執行上下文?

當控制器轉到可執行的代碼時,會進入該代碼對應的執行上下文,可以理解為該代碼對應的一個執行環境,就叫做執行上下文。

在JavaScript中運行環境有三種,分別是:

  • 全局環境:JavaScript代碼執行起來,首先就是進入全局環境。
  • 函數環境:當函數被調用執行時,就會進入函數中執行。
  • eval

所以在一個JavaScript程序中,就會產生多個不同的執行上下文,這時候就需要用到前面提到的數據結構來管理了,我們稱之為調用棧。當代碼在執行過程中,遇到上面說的三種情況,就會產生三種執行上下文,然后分別壓入調用棧中,等一個執行上下文執行完畢,彈出棧,才能執行下一個執行上下文中的代碼,這就是棧結構的特點。

執行上下文的特點

  • 單線程,其實javascript就是單線程,所以很好理解。
  • 同步執行,同步就是按順序,不能同時執行。
  • 全局上下文只有一個,它在瀏覽器關閉時才會彈出棧。
  • 函數的執行上下文的數目沒有限制。
  • 每次某個函數被調用時,就會有新的執行上下文,即使是調用的自身函數。
demo01
function f1(){
     var n = 999;
     function f2(){
         alert(n);
    }
    return f2;
}
var result = f1();
result();//999

我以上面這樣一個例子講解,執行上下文調用棧中的創建過程

image.png

執行上下文的生命周期

image.png

如圖所示,主要分為兩個階段,一個是創建階段,一個是執行階段

創建階段:
  • 生成變量對象,后面會講解
  • 建立作用域鏈
  • 確定this指向
執行階段:
  • 變量賦值
  • 函數引用
  • 執行其他代碼
執行完畢后彈棧,等待回收

變量對象和活動對象的區別就在于,執行周期不一樣,在創建階段叫做變量對象,在執行階段叫做活動對象。

變量對象

image.png

變量對象的創建主要有三個階段:

  • 1、創建arguments對象。
  • 2、檢查function函數聲明創建屬性。在VO對象中以函數名建立一個屬性,屬性值為函數的地址。如果函數名的屬性已經存在了,那么該屬性將會被新的引用所覆蓋。
  • 3、檢查var變量聲明創建屬性。在VO對象中以變量名建立一個屬性,屬性值為undefined。為了防止同名的屬性值會被修改為undefined,則會直接跳過,原屬性值不會被修改。

舉個變量提升和函數提升的例子,就明白了

demo02
function test(){
      console.log(a);
      console.log(foo());
      var a = 1;
      function foo(){
             return 2;
     }
}
test();

這是一個典型的變量提升和函數提升的例子,最后會輸出undefined和2,接下來以執行上下文的生命周期來講解,

創建過程
testEC = {
      VO:{},
      scopeChain:{},
      this:{}
}
VO = {
     arguments:{},
     foo:<foo reference>,
     a:undefined
}
執行階段
VO->AO
AO={
      arguments:{},
     foo:<foo reference>,
     a:2
}
等同于
function test(){
      function foo(){
           return 2;
     }
     var a;
     console.log(a);
     console.log(foo());
     a = 1;
}
test();

通過上面知識的講解,進一步了解到了變量提升和函數提升的底層原理,對后面知識的學習也做了鋪墊。

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 228,156評論 6 531
  • 序言:濱河連續發生了三起死亡事件,死亡現場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發現死者居然都...
    沈念sama閱讀 98,401評論 3 415
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 176,069評論 0 373
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 62,873評論 1 309
  • 正文 為了忘掉前任,我火速辦了婚禮,結果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 71,635評論 6 408
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發上,一...
    開封第一講書人閱讀 55,128評論 1 323
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,203評論 3 441
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 42,365評論 0 288
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當地人在樹林里發現了一具尸體,經...
    沈念sama閱讀 48,881評論 1 334
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 40,733評論 3 354
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發現自己被綠了。 大學時的朋友給我發了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 42,935評論 1 369
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,475評論 5 358
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質發生泄漏。R本人自食惡果不足惜,卻給世界環境...
    茶點故事閱讀 44,172評論 3 347
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 34,582評論 0 26
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 35,821評論 1 282
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 51,595評論 3 390
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 47,908評論 2 372

推薦閱讀更多精彩內容