摘要:執行上下文作用域鏈和內部機制一執行上下文執行上下文是代碼的執行環境,它包括的值變量對象和函數。創建作用域鏈一旦可變對象創建完,引擎就開始初始化作用域鏈。
執行上下文、作用域鏈和JS內部機制(Execution context, Scope chain and JavaScript internals)
一、執行上下文執行上下文(Execution context EC)是js代碼的執行環境,它包括this的值、變量、對象和函數。js執行上下文有3種類型
全局上下文是文件第一次加載到瀏覽器,js代碼開始執行的默認執行上下文。在瀏覽器環境中,嚴格模式下this的值為undefined,否則this的值為window對象。GEC只能有一個(因為js執行的全局環境只能有一個)。
函數執行時創建函數執行上下文,每個函數都有自己的執行上下文。FEC可以獲取到GEC中的內容。當在全局上下文中執行代碼時js引擎發現一個函數調用,則創建一個函數執行上下文。
執行eval時創建
二、執行上下文棧執行上下文棧Execution context stack (ECS)是執行js代碼時創建的執行棧結構。GEC默認在棧的最里層,當js引擎發現一個函數調用,則創建這個函數的FEC并push進棧,js引擎執行棧頂上下文關聯的函數,一旦函數執行完,則將其FEC pop出棧,并往下執行。
看個例子(動圖插不了棧動圖鏈接)
var a = 10; function functionA() { console.log("Start function A"); function functionB(){ console.log("In function B"); } functionB(); } functionA(); console.log("GlobalContext");
當上面的代碼在瀏覽器中加載時,js引擎先將GEC push入ECS中,當在GEC中調用functionA時,functionA執行上下文被push入棧,并開始執行functionA。
當functionB在functionA中被調用時,functionB的執行上下文被push入棧,開始執行functionB,當functionB中內容執行完,functionB執行上下文被pop出棧,此時棧頂為functionA的執行上下文,繼續執行functionA的代碼,執行完后pop出棧,棧頂為GEC。
最終執行GEC中代碼,執行完pop整個代碼結束。
上面討論了js引擎如何處理執行上下文(push和pop),下面討論js引擎如何創建執行上下文,這個過程分為兩個階段:創建階段和執行階段。
三、創建執行上下文 1. 創建階段(后面又叫編譯階段)js引擎調用函數,但函數還沒開始執行階段。
js引擎在這個階段對整個函數進行一個編譯(compile the code),主要干了下面三件事:
可變對象是包含所有變量、函數參數和內部函數聲明信息的特殊對象,它是一個特殊對象且沒有__proto__屬性。
一旦可變對象創建完,js引擎就開始初始化作用域鏈。作用域鏈是一個當前函數所在的可變對象的列表,其中包括GEC的可變對象和當前函數的可變對象。
初始化this的值
下面通過一個例子進行說明
function funA (a, b) { var c = 3; var d = 2; d = function() { return a - b; } } funA(3, 2);
當調用funA和執行funA前的這段時間,js引擎為funA創建了一個executionContextObj如下
executionContextObj = { variableObject: {}, // All the variable, arguments and inner function details of the funA scopechain: [], // List of all the scopes inside which the current function is this // Value of this }
可變對象包含參數對象(包含函數參數的細節),聲明的變量和函數,如下所示
variableObject = { argumentObject : { 0: a, 1: b, length: 2 }, a: 3, b: 2 c: undefined, d: undefined then pointer to the function defintion of d }
argumentObject如上所示
函數中的變量會被初始為undefined,參數也會在可變對象中呈現
如果變量在參數對象中已存在,js引擎選擇忽略
js引擎在當前函數中遇到函數定義,會用函數名創建一個屬性指向函數定義存儲的堆內容
2. 執行階段在此階段,js引擎會重掃一遍函數,用具體的變量的值來更新可變對象,并執行代碼內容。
執行階段執行完后,可變對象的值如下:
variableObject = { argumentObject : { 0: a, 1: b, length: 2 }, a: 3, b: 2, c: 3, d: undefined then pointer to the function defintion of d }四、完整的例子
代碼如下
a = 1; var b = 2; cFunc = function(e) { var c = 10; var d = 15; a = 3 function dFunc() { var f = 5; } dFunc(); } cFunc(10);全局編譯階段
當瀏覽器加載上面的代碼后,js引擎進入編譯階段,只處理聲明,不處理值。下面走讀一遍代碼:
a被賦值1,但它并不是個變量或函數聲明,js引擎在編譯階段什么都不做;
b變量聲明初始化為undefined;
cFunc函數聲明初始化為undefined。
此時的
globalExecutionContextObj = { variableObject: { // 原文中有時用activationObj argumentObj : { length:0 }, b: undefined, cFunc: Pointer to the function definition }, scopeChain: [GLobal execution context variable object], this: value of this }全局執行階段
再接著上面,js引擎進入執行階段并再過一遍。此時將會更新變量名和執行
js引擎發現可變對象中沒有a屬性,因此在GEC中添加a屬性,并初始化為1;
可變對象有b,直接更新b的值為2;
接著是函數聲明,不做任何事;
最后調用cFunc,js引擎再次進入編譯階段創建一個cFunc的執行上下文。
此時
globalExecutionContextObj = { variableObject: { argumentObj : { length:0 }, b: 2, cFunc: Pointer to the function definition, a: 1 }, scopeChain: [GLobal execution context variable object], this: value of this }cFunc的編譯階段
由于cFunc有個參數e,js引擎會在cFunc執行上下文對象可變對象添加e屬性,并初始化為2
js引擎查看cFunc執行上下文的可變對象沒有c,因此添加c,并初始化為undefined,d類似;
a = 3非聲明,跳過;
函數聲明,創建dFunc屬性指向函數的堆空間;
對dFunc執行語句忽略
此時
cFuncExecutionContextObj = { activationbj: { argumentObj : { 0: e, length:1 }, e: 10, c: undefined, d: undefined dFunc: Pointer to the function definition, }, scopeChain: [cFunc variable object, Global exection context variable object], this: value of this }cFunc的執行階段
c和d獲取到初始化值;
a不是cFunc執行上下文對象中的屬性,js引擎會在作用率鏈的幫助下轉到GEC(全局執行上下文),查找a是否在GEC中。如果不存在,則會在當前作用域創建并初始化它;如果GEC中有,則更新其值,這里會更新為3。js引擎只有在發現一個變量在當前執行上下文對象屬性中找不到時會跳轉到GEC中;
創建dFunc屬性并指向函數的堆內存
cFuncExecutionContextObj = { activationbj: { argumentObj : { 0: e, length:1 }, e: 10, c: 10, d: 15 dFunc: Pointer to the function definition, }, scopeChain: [cFunc variable object, Global exection context variable object], this: value of this }
調用dFunc,js引擎再次進入編譯階段,創建dFunc執行上下文對象。
dFunc執行上下文對象可以訪問到cFunc和全局作用域中的所有變量和函數;同樣cFunc可以訪問到全局的,但不能訪問dFunc中的;全局上下文對象不能訪問cFunc和dFunc中的變量和對象。
有了上面的概念,對hoisting(變量提升)應該更容易理解了。
作用域鏈是當前函數所在的可變對象列表
看下面一段代碼
a = 1; var b = 2; cFunc = function(e) { var c = 10; var d = 15; console.log(c); console.log(a); function dFunc() { var f = 5; console.log(f) console.log(c); console.log(a); } dFunc(); } cFunc(10);
當cFunc被調用時,cFunc的作用域鏈如下
Scope chain of cFunc = [ cFunc variable object, Global Execution Context variable object]
當dFunc被調用時,dFunc在cFunc中,dFunc的作用域鏈包含dFunc、cFunc和全局可變對象
Scope chain of dFunc = [dFunc variable object, cFunc variable object, Global execution context variable object]
當我們嘗試訪問dFunc中的f,js引擎查看f是否可從dFunc的可變對象中獲取,找到console輸出;
訪問c變量,js引擎首先在dFunc的可變對象中獲取,不能獲取,則到cFunc的可變對象中去獲取,找到console輸出;
訪問a變量,同上,最后找到GEC的可變對象,獲取到并console輸出
同樣,cFunc中獲取c和a類似
在cFunc中訪問不到f變量,但dFunc中可以通過作用域鏈獲取到c和d
文章版權歸作者所有,未經允許請勿轉載,若此文章存在違規行為,您可以聯系管理員刪除。
轉載請注明本文地址:http://m.specialneedsforspecialkids.com/yun/97581.html
摘要:前言這段時間一直在消化作用域鏈和閉包的相關知識。而作用域鏈則是這套規則這套規則的具體運行。是變量對象的縮寫那這樣放有什么好處呢我們知道作用域鏈保證了當前執行環境對符合訪問權限的變量和函數的有序訪問。 前言:這段時間一直在消化作用域鏈和閉包的相關知識。之前看《JS高程》和一些技術博客,對于這些概念的論述多多少少不太清楚或者不太完整,包括一些大神的技術文章。這也給我的學習上造成了一些困惑,...
摘要:本期推薦文章從作用域鏈談閉包,由于微信不能訪問外鏈,點擊閱讀原文就可以啦。推薦理由這是一篇譯文,深入淺出圖解作用域鏈,一步步深入介紹閉包。作用域鏈的頂端是全局對象,在全局環境中定義的變量就會綁定到全局對象中。 (關注福利,關注本公眾號回復[資料]領取優質前端視頻,包括Vue、React、Node源碼和實戰、面試指導) 本周開始前端進階的第二期,本周的主題是作用域閉包,今天是第6天。 本...
摘要:閉包面試題解由于作用域鏈機制的影響,閉包只能取得內部函數的最后一個值,這引起的一個副作用就是如果內部函數在一個循環中,那么變量的值始終為最后一個值。 (關注福利,關注本公眾號回復[資料]領取優質前端視頻,包括Vue、React、Node源碼和實戰、面試指導) 本周正式開始前端進階的第二期,本周的主題是作用域閉包,今天是第8天。 本計劃一共28期,每期重點攻克一個面試重難點,如果你還不了...
摘要:使用上一篇文章的例子來說明下自由變量進階期深入淺出圖解作用域鏈和閉包訪問外部的今天是今天是其中既不是參數,也不是局部變量,所以是自由變量。 (關注福利,關注本公眾號回復[資料]領取優質前端視頻,包括Vue、React、Node源碼和實戰、面試指導) 本周正式開始前端進階的第二期,本周的主題是作用域閉包,今天是第7天。 本計劃一共28期,每期重點攻克一個面試重難點,如果你還不了解本進階計...
閱讀 3001·2021-10-13 09:39
閱讀 2697·2021-09-27 13:34
閱讀 2036·2019-08-30 15:55
閱讀 3266·2019-08-30 15:43
閱讀 3641·2019-08-30 11:16
閱讀 1756·2019-08-26 18:28
閱讀 1293·2019-08-26 13:56
閱讀 918·2019-08-26 13:35