摘要:為啥因為變量提升,變量的聲明被提升到當前作用域的頂部了。也就是可以想象成這樣此外,還有函數(shù)提升,和變量提升類似和被提升了,所以不會報錯。開始處理函數(shù)聲明,再次提醒,函數(shù)表達式不會被處理。
變量提升 & 函數(shù)提升
function f() { console.log(a); // ? var a = 1; } f();
簡單簡單,打印結果是 undefined。為啥?因為變量提升,變量 a 的聲明被提升到當前作用域的頂部了。也就是可以想象成這樣:
function f() { var a; console.log(a); a = 1; }
此外,還有函數(shù)提升,和變量提升類似:
f1(); f2(); function f1() { console.log(1) } function f2() { console.log(2) }
f1 和 f2 被提升了,所以不會報錯。很多文章中都會經(jīng)常提到提升這個概念。
但是,這是真的嗎?JS 解釋器在執(zhí)行代碼時還順帶幫你變動一下代碼的順序?想想也覺得不太可能嘛~那么到底是咋回事捏?
JavaScript 的可執(zhí)行代碼(executable code)有三種,分別為 全局代碼、函數(shù)代碼以及 eval 代碼。鑒于 eval 不被推薦使用,咱就不理它。
執(zhí)行上下文當執(zhí)行全局代碼時,會創(chuàng)建一個全局執(zhí)行上下文。當執(zhí)行一個函數(shù)的時候,會創(chuàng)建一個函數(shù)執(zhí)行上下文。全局執(zhí)行上下文只會有一個,而函數(shù)執(zhí)行上下文可以有很多很多個。JS 引擎會創(chuàng)建 執(zhí)行上下文棧(execution context stack, ECS)去管理這些執(zhí)行上下文。我們以函數(shù)執(zhí)行上下文為例。
變量對象對于一個執(zhí)行上下文而言,可以抽象化為這樣一個對象:
contextObject = { VariableObject, ScopeChain, thisValue }
其中變量對象(Variable Object,VO)是包含與當前執(zhí)行上下文相關的數(shù)據(jù)作用域,它存儲著當前上下文中定義的變量聲明、函數(shù)聲明(注意:不包含函數(shù)表達式),另外函數(shù)執(zhí)行上下文中還會有參數(shù)。
變量對象是一個抽象概念,在不同的執(zhí)行上下文中會以不同的對象呈現(xiàn)。比如在全局上下文中,變量對象就是全局對象本身(這也是為什么我們能夠通過全局對象的屬性名稱引用全局變量)。而對于函數(shù)執(zhí)行上下文,是用活動對象(Activation Object,AO)來表示變量對象的,我們訪問的便是活動對象上的屬性。
執(zhí)行過程執(zhí)行上下文中的代碼會被分出兩個階段進行處理,分別為:
進入執(zhí)行上下文,即分析創(chuàng)建階段
執(zhí)行代碼,即執(zhí)行階段
分析創(chuàng)建階段此時進入執(zhí)行上下文,但在開始執(zhí)行代碼之前。此時的變量對象:
如果是函數(shù)上下文,那么會通過函數(shù)的 arguments 屬性初始化,并處理形參。此時變量對象中會包含一個 arguments 對象,以及所有的形參,并且會檢查是否有與之對應的實參,有則值初始化為實參的值,沒有的話就設為 undefined。
處理函數(shù)聲明。開始處理函數(shù)聲明,再次提醒,函數(shù)表達式不會被處理。此時會檢查變量對象中是否已經(jīng)有同名屬性,如果有,則替換它,如果沒有則創(chuàng)建屬性,值初始化為對應的函數(shù)對象。
處理變量聲明。與處理函數(shù)聲明不同的是,如果變量對象中存在有同名屬性,此時會跳過該變量不做任何事,即不會產(chǎn)生覆蓋,如果沒有則創(chuàng)建屬性并初始化為 undefined。
了解了第一個階段,那么來看個例子:
function f(a, b) { var c = 3; function d() {}; var e = function() {}; (function f() {}); } f(1, 2);
當執(zhí)行 f 函數(shù)時,進入執(zhí)行上下文,我們可以描述出此時的 AO:
AO = { arguments: { 0: 1, 1: 2, length: 2 }, a: 1, b: 2, c: undefined, d: reference to function() {}, e: undefined, } // 注意:(function f() {}) 為函數(shù)表達式,不會被處理執(zhí)行階段
執(zhí)行階段就相對簡單,代碼順序執(zhí)行,并且會根據(jù)代碼去修改變量對象中的值,所以當函數(shù)執(zhí)行完后的 AO 會是:
AO = { arguments: { 0: 1, 1: 2, length: 2 }, a: 1, b: 2, c: 3, d: reference to function() {}, e: reference to function() {}, }總結
綜上所述:
全局上下文的變量對象即是全局對象;
函數(shù)上下文的變量對象會以 arguments 對象初始化;
在第一個階段會依次給變量對象添加形參(函數(shù)上下文)、函數(shù)聲明、變量聲明;
在第二個階段,會修改變量對象中的屬性值。
思考題console.log(f); // ? function f() {} var f = 1;
打印結果會是一個函數(shù)對象。因為第一個階段中先處理函數(shù)聲明后處理變量聲明,當處理變量聲明時變量對象中已經(jīng)存在同名屬性 f,不會做任何事直接跳過,所以 f 的值會是一個函數(shù)。
你也可以多找?guī)椎李}嘗試分析以加深理解。當然,日常工作中還是直接想成提升來得方便些,只是希望通過本文能讓你了解到一些背后的事情。最后,如果文中有任何錯誤或不足之處,還請指出~
文章版權歸作者所有,未經(jīng)允許請勿轉載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉載請注明本文地址:http://systransis.cn/yun/104225.html
摘要:瀏覽器的主要組成包括有調用堆棧,事件循環(huán),任務隊列和。好了,現(xiàn)在有了前面這些知識,我們可以看一下這道題的講解過程實現(xiàn)步驟調用會將函數(shù)放入調用堆棧。由于調用堆棧是空的,事件循環(huán)將選擇回調并將其推入調用堆棧進行處理。進程再次重復,堆棧不會溢出。 JavaScript是前端開發(fā)中非常重要的一門語言,瀏覽器是他主要運行的地方。JavaScript是一個非常有意思的語言,但是他有很多一些概念,大...
摘要:中國的是一個陰謀讓我們首先回到的初衷。春陽曾經(jīng)分享過的藏寶圖報告里有過一個關于家廠商毛利水平的統(tǒng)計,如下圖所示,其中位數(shù)是。每一年,都會有人問我,春陽,你覺得SaaS行業(yè)到時候了嗎?每一年,都會有媒體發(fā)文,SaaS已來,未來可期....是的,每一年...行業(yè)的媒體人喜歡給SaaS灌雞湯是沒有毛病的,本身這就是個留不住人才、熬不出日子的行業(yè),如果我們再看衰它,媒體本身也是活不下去了…對這個問題...
摘要:類就是產(chǎn)生各種不同類型事件的產(chǎn)出器,比如定時器事件讀寫事件等等,為了提升民族榮譽感,我們將這些各種事件比作各種戰(zhàn)斗機比如殲殲和殲。類就相對容易介入了,這玩意顯然就是一個航空母艦了,為了提升民族榮譽感,我們就把類當作是遼寧艦。 [原文地址:https://blog.ti-node.com/blog...] 實際上php.net上是有event擴展的使用說明手冊,但是呢,對于初學者來說卻并...
閱讀 1686·2021-11-19 09:40
閱讀 2939·2021-09-24 10:27
閱讀 3227·2021-09-02 15:15
閱讀 1888·2019-08-30 15:54
閱讀 1213·2019-08-30 15:54
閱讀 1377·2019-08-30 13:12
閱讀 642·2019-08-28 18:05
閱讀 2808·2019-08-27 10:53