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

資訊專欄INFORMATION COLUMN

JavaScript 之執(zhí)行上下文

FrozenMap / 2172人閱讀

摘要:本文就梳理有關(guān)執(zhí)行上下文也叫執(zhí)行環(huán)境的知識。全局代碼的執(zhí)行上下文棧可以表示為函數(shù)代碼當(dāng)執(zhí)行函數(shù)代碼時,函數(shù)代碼上下文被壓入到執(zhí)行上下文棧中。

本文共 1090 字,讀完只需 4 分鐘
概述

JavaScript 是函數(shù)式編程語言,作用域也是以函數(shù)為單位,那么,這些函數(shù)代碼塊是怎么樣的順序進(jìn)行的呢, JS 的可執(zhí)行代碼又分為 3 種,不同類型的代碼有不一樣的執(zhí)行環(huán)境。本文就梳理有關(guān) JS 執(zhí)行上下文(execution context),也叫執(zhí)行環(huán)境的知識。

活動的執(zhí)行代碼的上下文在語言底層邏輯上構(gòu)成一個執(zhí)行上下文棧(excution context stack),我們知道,棧有兩個行為,壓入棧和彈出棧,并且有后進(jìn)先出的特點, 最先進(jìn)入的項會在棧底最后彈出。

不同執(zhí)行環(huán)境的有其相應(yīng)的變量對象(Variable Object),某個執(zhí)行環(huán)境的所有可執(zhí)行代碼都執(zhí)行完畢后,該環(huán)境中的變量對象和函數(shù)定義會被清除。

函數(shù)代碼會在執(zhí)行完后清除變量占用的內(nèi)存,全局代碼則會在關(guān)閉環(huán)境,比如關(guān)閉瀏覽器后清除。

一、代碼類型

JS 中可執(zhí)行的代碼可分為三種類型:

全局代碼

函數(shù)代碼

Eval 代碼

全局代碼

在 web 瀏覽器中,全局執(zhí)行環(huán)境是 window 對象,所有的全局變量和函數(shù)都是作為 window 的屬性和方法存在。

全局代碼的執(zhí)行上下文??梢员硎緸椋?/p>

ECStack = [
    globalContext
]

函數(shù)代碼

當(dāng)執(zhí)行函數(shù)代碼時,函數(shù)代碼上下文被壓入到執(zhí)行上下文棧中。函數(shù)代碼的執(zhí)行環(huán)境中,有自己的內(nèi)部的定義的變量和聲明。

function foo1() {
    var name1 = "a";
    console.log(name1)
}

function foo2(){
    var name2 = "b";
    console.log(name2)
}

foo1();
foo2();

上面的執(zhí)行上下文可以表示為:

ECStack = [
    globalContext
];

ECStack.push( functionContext);
ECStack.pop();
ECStack.push( functionContext);
ECStack.pop();

一個函數(shù),可能有多個執(zhí)行上下文,每個函數(shù)的調(diào)用都會產(chǎn)生新的上下文。

function foo() {
    ...
}

foo("a");
foo("b");
foo("c");

eval 代碼

eval 關(guān)鍵字接受一個字符串作為參數(shù),并將其作為書寫文字上下文的代碼。

function foo(str, a) {
    eval( str );  // 聲明一個新變量
    console.log(a, b);
}
var b = 123;

foo("var b = 456 ", 123) // 123, 456

以上代碼引用《你不知道的 JavaScript》的例子,eval 函數(shù)中的字符串,被當(dāng)做可執(zhí)行代碼,最后在 foo 函數(shù)中聲明了 b 變量,并覆蓋了 foo 函數(shù)外部的 b 變量。

eval 函數(shù)和 JS 的詞法作用域的行為,會造成變量環(huán)境和執(zhí)行上下文的混亂,盡量別使用它。

二、執(zhí)行上下文棧

前面其實已經(jīng)提到很多關(guān)于執(zhí)行上下文棧的內(nèi)容,簡而言之,JS 在遇到全局代碼,函數(shù)代碼,eval 代碼時,會創(chuàng)建相應(yīng)的執(zhí)行上下文,不同的上下文,有其變量對象(variable object: VO)和作用域。

看這一段代碼:

function func3() {
    console.log("fun3")
}

function func2() {
    func3();
}

function func1() {
    func2();
}

func1();

以上代碼引用自冴羽的github,代碼順序用執(zhí)行上下文棧來表示就是:

ECStack.push(globalContext);

// 調(diào)用 func1(), 進(jìn)入 func1 執(zhí)行環(huán)境
ECStack.push( functionContext);

// func1中調(diào)用了 func2,進(jìn)入 func2 的執(zhí)行上下文
ECStack.push( functionContext);

// func2 調(diào)用 func3, 進(jìn)入 func3 的執(zhí)行上下文
ECStack.push( functionContext);

// func3 執(zhí)行完畢,退出 func3 執(zhí)行環(huán)境
ECStack.pop();

// func2 執(zhí)行完畢,退出 func2 執(zhí)行環(huán)境
ECStack.pop();

// func1 執(zhí)行完畢,退出 func1 執(zhí)行環(huán)境
ECStack.pop();

函數(shù)代碼執(zhí)行完畢后,執(zhí)行上下文棧底只剩下全局代碼上下文 globalContext, 直到關(guān)閉瀏覽器才會徹底清空執(zhí)行上下文棧。

總結(jié)

代碼在執(zhí)行時會創(chuàng)建由不同作用域構(gòu)成的作用域鏈,作用域鏈保證 JS 中的變量和函數(shù)都能夠有序地訪問和執(zhí)行。

如果是函數(shù)代碼,則將其 活動對象 作為變量對象,活動對象最開始時只包含一個對象,即 arguments 對象。

后面的文章會介紹變量對象(VO)和作用域鏈的具體內(nèi)容,敬請期待吧。

博客內(nèi)容源自個人公眾號,專注分享原創(chuàng)文章,歡迎關(guān)注。

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

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

相關(guān)文章

  • JavaScript深入執(zhí)行下文

    摘要:深入系列第七篇,結(jié)合之前所講的四篇文章,以權(quán)威指南的為例,具體講解當(dāng)函數(shù)執(zhí)行的時候,執(zhí)行上下文棧變量對象作用域鏈?zhǔn)侨绾巫兓?。前言在深入之?zhí)行上下文棧中講到,當(dāng)代碼執(zhí)行一段可執(zhí)行代碼時,會創(chuàng)建對應(yīng)的執(zhí)行上下文。 JavaScript深入系列第七篇,結(jié)合之前所講的四篇文章,以權(quán)威指南的demo為例,具體講解當(dāng)函數(shù)執(zhí)行的時候,執(zhí)行上下文棧、變量對象、作用域鏈?zhǔn)侨绾巫兓摹?前言 在《Jav...

    gougoujiang 評論0 收藏0
  • JavaScript深入作用域鏈

    摘要:下面,讓我們以一個函數(shù)的創(chuàng)建和激活兩個時期來講解作用域鏈?zhǔn)侨绾蝿?chuàng)建和變化的。這時候執(zhí)行上下文的作用域鏈,我們命名為至此,作用域鏈創(chuàng)建完畢。 JavaScript深入系列第五篇,講述作用鏈的創(chuàng)建過程,最后結(jié)合著變量對象,執(zhí)行上下文棧,讓我們一起捋一捋函數(shù)創(chuàng)建和執(zhí)行的過程中到底發(fā)生了什么? 前言 在《JavaScript深入之執(zhí)行上下文?!分兄v到,當(dāng)JavaScript代碼執(zhí)行一段可執(zhí)行代...

    waltr 評論0 收藏0
  • JavaScript深入閉包

    摘要:深入系列第八篇,介紹理論上的閉包和實踐上的閉包,以及從作用域鏈的角度解析經(jīng)典的閉包題。定義對閉包的定義為閉包是指那些能夠訪問自由變量的函數(shù)。 JavaScript深入系列第八篇,介紹理論上的閉包和實踐上的閉包,以及從作用域鏈的角度解析經(jīng)典的閉包題。 定義 MDN 對閉包的定義為: 閉包是指那些能夠訪問自由變量的函數(shù)。 那什么是自由變量呢? 自由變量是指在函數(shù)中使用的,但既不是函數(shù)參數(shù)也...

    caige 評論0 收藏0
  • JavaScript深入執(zhí)行下文

    摘要:深入系列第三篇,講解執(zhí)行上下文棧的是如何執(zhí)行的,也回答了第二篇中的略難的思考題。 JavaScript深入系列第三篇,講解執(zhí)行上下文棧的是如何執(zhí)行的,也回答了第二篇中的略難的思考題。 順序執(zhí)行? 如果要問到 JavaScript 代碼執(zhí)行順序的話,想必寫過 JavaScript 的開發(fā)者都會有個直觀的印象,那就是順序執(zhí)行,畢竟: var foo = function () { ...

    codecraft 評論0 收藏0
  • JavaScript專題遞歸

    摘要:專題系列第十八篇,講解遞歸和尾遞歸定義程序調(diào)用自身的編程技巧稱為遞歸。然而非尾調(diào)用函數(shù),就會創(chuàng)建多個執(zhí)行上下文壓入執(zhí)行上下文棧。所以我們只用把階乘函數(shù)改造成一個尾遞歸形式,就可以避免創(chuàng)建那么多的執(zhí)行上下文。 JavaScript 專題系列第十八篇,講解遞歸和尾遞歸 定義 程序調(diào)用自身的編程技巧稱為遞歸(recursion)。 階乘 以階乘為例: function factorial(n...

    asoren 評論0 收藏0
  • 【進(jìn)階1-2期】JavaScript深入執(zhí)行下文棧和變量對象

    摘要:本計劃一共期,每期重點攻克一個面試重難點,如果你還不了解本進(jìn)階計劃,點擊查看前端進(jìn)階的破冰之旅本期推薦文章深入之執(zhí)行上下文棧和深入之變量對象,由于微信不能訪問外鏈,點擊閱讀原文就可以啦。 (關(guān)注福利,關(guān)注本公眾號回復(fù)[資料]領(lǐng)取優(yōu)質(zhì)前端視頻,包括Vue、React、Node源碼和實戰(zhàn)、面試指導(dǎo)) 本周正式開始前端進(jìn)階的第一期,本周的主題是調(diào)用堆棧,今天是第二天。 本計劃一共28期,每期...

    Richard_Gao 評論0 收藏0

發(fā)表評論

0條評論

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