成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

理解 JavaScript 執(zhí)行棧

yacheng / 1340人閱讀

摘要:執(zhí)行棧所有的代碼在運行時都是在執(zhí)行上下文中進行的。引擎執(zhí)行棧頂?shù)暮瘮?shù),執(zhí)行完畢,彈出當(dāng)前執(zhí)行上下文。但是這里我們也可以用執(zhí)行棧來解釋。

這是 JavaScript 系列的第 3 篇。

引例

首先來看一個引例:

function foo() {
  console.log("1");
  bar();
  console.log("3");
}

function bar() {
  console.log("2");
}

foo();

這段代碼將從上往下依次執(zhí)行,并輸出 "1", "2", "3"。

我們可以看到,bar 函數(shù)的執(zhí)行順序似乎和它定義的順序沒有關(guān)系。為什么呢?這你就得弄懂執(zhí)行棧了。

執(zhí)行棧

所有的 JS 代碼在運行時都是在執(zhí)行上下文中進行的。執(zhí)行上下文是一個抽象的概念,JS 中有三種執(zhí)行上下文:

全局執(zhí)行上下文,默認(rèn)的,在瀏覽器中是 window 對象,并且 this 在非嚴(yán)格模式下指向它。

函數(shù)執(zhí)行上下文,JS 的函數(shù)每當(dāng)被調(diào)用時會創(chuàng)建一個上下文。

Eval 執(zhí)行上下文,eval 函數(shù)會產(chǎn)生自己的上下文,這里不討論。

通常,我們的代碼中都不止一個上下文,那這些上下文的執(zhí)行順序應(yīng)該是怎樣的?從上往下依次執(zhí)行?

棧,是一種數(shù)據(jù)結(jié)構(gòu),具有先進后出的原則。JS 中的執(zhí)行棧就具有這樣的結(jié)構(gòu),當(dāng)引擎第一次遇到 JS 代碼時,會產(chǎn)生一個全局執(zhí)行上下文并壓入執(zhí)行棧,每遇到一個函數(shù)調(diào)用,就會往棧中壓入一個新的上下文。引擎執(zhí)行棧頂?shù)暮瘮?shù),執(zhí)行完畢,彈出當(dāng)前執(zhí)行上下文。

以引例來說明。當(dāng) foo() 函數(shù)被調(diào)用,將 foo 函數(shù)的執(zhí)行上下文壓入執(zhí)行棧,接著執(zhí)行輸出 ‘1’;當(dāng) bar() 函數(shù)被調(diào)用,將 bar 函數(shù)的執(zhí)行上下文壓入執(zhí)行棧,接著執(zhí)行輸出 ‘2’;bar() 執(zhí)行完畢,被彈出執(zhí)行棧,foo() 函數(shù)接著執(zhí)行,輸出 ‘3’;foo() 函數(shù)執(zhí)行完畢,被彈出執(zhí)行棧。

那現(xiàn)在來看這個例子:

var count = 0;
function foo(count) {
  count += 1;
  console.log(count);
}
foo(count); // 1
foo(count); // 1

我們用執(zhí)行棧來理解一下,函數(shù)每次被調(diào)用都會產(chǎn)生新的執(zhí)行上下文,并被壓入執(zhí)行棧,執(zhí)行完畢后當(dāng)前上下文就會被彈出執(zhí)行棧。所以第一次調(diào)用應(yīng)該返回 1,第二次調(diào)用也應(yīng)該返回 1,第 n 次調(diào)用都應(yīng)該返回 1。

你理解了嗎?那再來看一個例子:

var count = 0;
function foo() {
  count += 1;
  console.log(count);
}
foo(count); // 1
foo(count); // 2

WTF?這個例子和上一個的區(qū)別是這里 foo 函數(shù)沒有指定形參。而這個例子其實就是通常說的函數(shù)內(nèi)部沒有使用 var 聲明的變量,都會被當(dāng)做全局變量(非嚴(yán)格模式)。

但是這里我們也可以用執(zhí)行棧來解釋。

函數(shù)的形參屬于函數(shù)執(zhí)行上下文,所以當(dāng)指定這個形參后,它就隨著函數(shù)被調(diào)用而新建,隨著函數(shù)銷毀而銷毀。如果不指定這個形參,上一篇文章已經(jīng)介紹過作用域鏈的概念,就會沿著作用域鏈找到全局變量 count,它屬于全局執(zhí)行上下文,這個時候再去調(diào)用 foo() 函數(shù)就會讀寫這個全局變量。

每個 foo() 函數(shù)調(diào)用后,給 count 加一,然后被彈出執(zhí)行棧,而全局執(zhí)行上下文的生命周期將伴隨著整個程序,所以第一次調(diào)用打印 1,第二次調(diào)用打印 2,第 n 次調(diào)用打印 n。

是不是很奇妙呢?隨著學(xué)習(xí)的深入,你會發(fā)現(xiàn) JavaScript 的奇妙遠(yuǎn)不止于此。

小結(jié)

執(zhí)行棧屬于 JavaScript 中基礎(chǔ)的概念,它與作用域、作用域鏈、執(zhí)行上下文、變量對象/活動對象的聯(lián)系都非常緊密。

文章首發(fā)于微信公眾號,理解 JavaScript 執(zhí)行棧

歡迎關(guān)注我的公眾號 cameraee,一起交流學(xué)習(xí)。

JavaScript 系列文章

分析 JavaScript 的數(shù)據(jù)類型與變量

理解 JavaScript 作用域

正在更新...

文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/99948.html

相關(guān)文章

  • 理解異步JavaScript

    摘要:當(dāng)函數(shù)結(jié)束,將會被從調(diào)用棧移出。事件循環(huán)事件循環(huán)的責(zé)任就是查看調(diào)用棧并確定調(diào)用棧是否為空。事件循環(huán)會再次檢查調(diào)用棧是否為空,如果為空的話,它會把事件回調(diào)壓入棧中,然后回調(diào)函數(shù)則被執(zhí)行。 寫在文章前 這篇文章是翻譯自Sukhjinder Arora的Understanding Asynchronous JavaScript。這篇文章描述了異步和同步JavaScript是如何在運行環(huán)境中,...

    ixlei 評論0 收藏0
  • 什么是JavaScript 事件循環(huán) ?

    摘要:此事件隊列的美妙之處在于它只是函數(shù)等待被調(diào)用和移動到調(diào)用棧的一個臨時存放區(qū)域。在事件循環(huán)不斷監(jiān)視調(diào)用棧是否為空現(xiàn)在確實是空的時候調(diào)用創(chuàng)建一個新的調(diào)用棧來執(zhí)行代碼。在執(zhí)行完之后進入了一個新的狀態(tài)這個狀態(tài)調(diào)用棧為空事件記錄表為空事件隊列也為空。 這篇文章是對個人認(rèn)為講解 JavaScript 事件循環(huán)比較清楚的一篇英文文章的簡單翻譯,原文地址是http://altitudelabs.com...

    tracymac7 評論0 收藏0
  • 理解javascript中的事件循環(huán)(Event Loop)

    摘要:主線程會暫時存儲等異步操作,直接向下執(zhí)行,當(dāng)某個異步事件觸發(fā)時,再通知主線程執(zhí)行相應(yīng)的回調(diào)函數(shù),通過這種機制,避免了單線程中異步操作耗時對后續(xù)任務(wù)的影響。 背景 在研究js的異步的實現(xiàn)方式的時候,發(fā)現(xiàn)了JavaScript 中的 macrotask 和 microtask 的概念。在查閱了一番資料之后,對其中的執(zhí)行機制有所了解,下面整理出來,希望可以幫助更多人。 先了解一下js的任務(wù)執(zhí)...

    mykurisu 評論0 收藏0
  • 深入理解JavaScript執(zhí)行上下文和執(zhí)行

    摘要:執(zhí)行上下文和執(zhí)行棧是中關(guān)鍵概念之一,是難點之一。理解執(zhí)行上下文和執(zhí)行棧同樣有助于理解其他的概念如提升機制作用域和閉包等。函數(shù)執(zhí)行完成,函數(shù)的執(zhí)行上下文出棧,并且被銷毀。 前言 如果你是一名 JavaScript 開發(fā)者,或者想要成為一名 JavaScript 開發(fā)者,那么你必須知道 JavaScript 程序內(nèi)部的執(zhí)行機制。執(zhí)行上下文和執(zhí)行棧是JavaScript中關(guān)鍵概念之一,是Ja...

    silenceboy 評論0 收藏0
  • 深入理解JavaScript執(zhí)行上下文和執(zhí)行

    摘要:執(zhí)行上下文和執(zhí)行棧是中關(guān)鍵概念之一,是難點之一。理解執(zhí)行上下文和執(zhí)行棧同樣有助于理解其他的概念如提升機制作用域和閉包等。函數(shù)執(zhí)行完成,函數(shù)的執(zhí)行上下文出棧,并且被銷毀。 前言 如果你是一名 JavaScript 開發(fā)者,或者想要成為一名 JavaScript 開發(fā)者,那么你必須知道 JavaScript 程序內(nèi)部的執(zhí)行機制。執(zhí)行上下文和執(zhí)行棧是JavaScript中關(guān)鍵概念之一,是Ja...

    leiyi 評論0 收藏0

發(fā)表評論

0條評論

最新活動
閱讀需要支付1元查看
<