摘要:技術(shù)上來說這個機制被稱為動態(tài)分配或代理。定義類一個類是一個正式的抽象集,它規(guī)定了對象的初始狀態(tài)和行為。技術(shù)上來說一個類表示構(gòu)造函數(shù)原型的組合。因此構(gòu)造函數(shù)創(chuàng)建對象并自動設(shè)置新創(chuàng)建實例的原型。第二次調(diào)用時,相同的上下文再次被壓入棧并恢復(fù)。
原文:JavaScript. The Core: 2nd Edition
作者:Dmitry Soshnikov
文章其他語言版本:俄語
這篇文章是 JavaScript. The Core 演講的第二版,文章內(nèi)容專注于 ECMAScript 編程語言和其運行時系統(tǒng)的核心組件。
面向讀者:有經(jīng)驗的開發(fā)者、專家
文章第一版 涵蓋了 JS 語言通用的方面,該文章描述的抽象大多來自古老的 ES3 規(guī)范,也引用了一些 ES5 和 ES6( ES2015 )的變更。
從 ES2015 開始,規(guī)范更改了一些核心組件的描述和結(jié)構(gòu),引入了新的模型等等。所以這篇文章我將聚焦新的抽象,更新的術(shù)語和在規(guī)范版本更替中仍然維護并保持一致的非?;镜?JS 結(jié)構(gòu)。
文章涵蓋 ES2017+ 運行時系統(tǒng)的內(nèi)容。
注釋:最新 ECMAScript 規(guī)范 版本可以在 TC-39 網(wǎng)站上查看。
我將從對象的概念開始講起,它是 ECMAScript 的根本。
對象ECMAScript 是一門面向?qū)ο?、基于原型進行組織的編程語言,且它的核心抽象為對象的概念。
定義1:對象:對象是屬性的集合并且有一個原型(prototype)對象。原型的值為一個對象或 null 。
我們來看一個基本的對象示例。對象的原型可通過內(nèi)部的 [[Prototype]] 屬性引用,在用戶代碼層面則是暴露在 __proto__ 屬性上。
代碼如下:
let point = { x: 10, y: 20, };
上面的對象有兩個顯式的屬性和一個隱藏的 __proto__ 屬性,它是 point 對象的原型引用:
注:對象也可能存儲 symbol 。閱讀這篇文章了解更多關(guān)于 symbol 的內(nèi)容。
原型對象用于實現(xiàn)動態(tài)分配機制的繼承。我們先思考一下原型鏈概念,以便詳細了解這個機制。
原型所有對象在創(chuàng)建的時候都會得到原型。如果沒有顯式地設(shè)置原型,那么對象接收默認原型作為它們的繼承對象。
定義2:原型:原型是一個代理對象,用來實現(xiàn)基于原型的繼承。
原型可以通過 __proto__ 屬性或 Object.create 方法顯式的設(shè)置。
// Base object. let point = { x: 10, y: 20, }; // Inherit from `point` object. let point3D = { z: 30, __proto__: point, }; console.log( point3D.x, // 10, inherited point3D.y, // 20, inherited point3D.z // 30, own );
注:默認情況下,對象接收 Object.prototype 作為它們的繼承對象。
任何對象都可作為其它對象的原型,且原型本身可以有原型。如果對象的原型不為 null ,原型的原型不為 null ,以此類推,這就叫做原型鏈。
定義3:原型鏈:原型鏈?zhǔn)菍ο蟮挠邢捩溄?,用來實現(xiàn)繼承和共享屬性。
規(guī)則非常簡單:如果對象自身沒有一個屬性,就會試圖在原型上解析屬性,然后原型的原型,直到查找完整個原型鏈。
技術(shù)上來說這個機制被稱為動態(tài)分配或代理。
定義4:代理:一個在繼承鏈上解析屬性的機制。這個過程是在運行時發(fā)生的,因此也被叫做動態(tài)分配。
注:與此相反的靜態(tài)分配是在編譯的時候解析引用的,動態(tài)分配則是在運行時。
如果屬性最終都沒有在原型鏈上找到的話,那么返回 undefined 值。
// An "empty" object. let empty = {}; console.log( // function, from default prototype empty.toString, // undefined empty.x, );
從上面的代碼可以知道,一個默認的對象實際上永遠不為空--它總是從 Object.prototype 繼承一些東西。如果想要創(chuàng)建一個無原型的字典(dictionary),我們必須明確地將原型設(shè)為 null :
// Doesn"t inherit from anything. let dict = Object.create(null); console.log(dict.toString); // undefined
動態(tài)分配機制允許繼承鏈完全可變,提供修改代理對象的能力:
let protoA = {x: 10}; let protoB = {x: 20}; // Same as `let objectC = {__proto__: protoA};`: let objectC = Object.create(protoA); console.log(objectC.x); // 10 // Change the delegate: Object.setPrototypeOf(objectC, protoB); console.log(objectC.x); // 20
注:即使 __proto__ 現(xiàn)在是標(biāo)準(zhǔn)屬性,并且在解釋時使用易于理解,但實踐時傾向使用操作原型的 API 方法,如 Object.create、Object.getPrototypeOf、Object.setPrototypeOf ,類似于反射(Reflect)模塊。
從上面 Object.prototype 示例我們知道同一個原型可以給多個對象共享。從這個原理出發(fā),ECMAScript 實現(xiàn)了基于類的繼承。我們看下示例,并且深入了解 JS 的 “類(class)” 抽象。
類當(dāng)多個對象共享同一個初始的狀態(tài)和行為時,它們就形成了一個類。
定義5:類:一個類是一個正式的抽象集,它規(guī)定了對象的初始狀態(tài)和行為。
假如我們需要多個對象繼承同一個原型,我們當(dāng)然可以創(chuàng)建這個原型并顯式的繼承它:
// Generic prototype for all letters. let letter = { getNumber() { return this.number; } }; let a = {number: 1, __proto__: letter}; let b = {number: 2, __proto__: letter}; // ... let z = {number: 26, __proto__: letter}; console.log( a.getNumber(), // 1 b.getNumber(), // 2 z.getNumber(), // 26 );
我們可以從下圖看到這些關(guān)系:
然而這明顯很繁瑣。類抽象正是服務(wù)這個目的 - 作為一個語法糖(和構(gòu)造器在語義上所做的一樣,但是是更友好的語法形式),它讓我們使用更方便的模式創(chuàng)建那些對象:
class Letter { constructor(number) { this.number = number; } getNumber() { return this.number; } } let a = new Letter(1); let b = new Letter(2); // ... let z = new Letter(26); console.log( a.getNumber(), // 1 b.getNumber(), // 2 z.getNumber(), // 26 );
注: ECMAScript 中基于類的繼承是在基于原型的代理之上實現(xiàn)的。注:一個“類”只是理論上的抽象。技術(shù)上來說,它可以像 Java 或 C++ 一樣通過靜態(tài)分配來實現(xiàn),也可以像 JavaScript、Python、Ruby 一樣通過動態(tài)分配(代理)來實現(xiàn)。
技術(shù)上來說一個“類”表示“構(gòu)造函數(shù) + 原型”的組合。因此構(gòu)造函數(shù)創(chuàng)建對象并自動設(shè)置新創(chuàng)建實例的原型。這個原型存儲在
定義6:構(gòu)造器:構(gòu)造器是一個函數(shù),它用來創(chuàng)建實例并自動設(shè)置它們的原型。
我們可以顯式的使用構(gòu)造函數(shù)。此外,在類抽象引入之前,JS 開發(fā)者過去因為沒有更好的選擇而這樣做(我們依然可以在互聯(lián)網(wǎng)上找到大量這樣的遺留代碼):
function Letter(number) { this.number = number; } Letter.prototype.getNumber = function() { return this.number; }; let a = new Letter(1); let b = new Letter(2); // ... let z = new Letter(26); console.log( a.getNumber(), // 1 b.getNumber(), // 2 z.getNumber(), // 26 );
創(chuàng)建單級的構(gòu)造函數(shù)非常簡單,而從父類繼承的模式則要求更多的模板代碼。目前這些模板代碼作為實現(xiàn)細節(jié)被隱藏,而這正是在我們創(chuàng)建 JavaScript 類時在底層所發(fā)生的。
注:構(gòu)造函數(shù)就是類繼承的實現(xiàn)細節(jié)。
我們看一下對象和它們的類的關(guān)系:
上圖顯示了每個對象都有一個關(guān)聯(lián)的原型。就連構(gòu)造函數(shù)(類)也有原型也就是 Function.prototype 。我們看到 a、b 和 c 是 Letter 的實例,它們的原型是 Letter.prototype 。
注:所有對象的實際原型總是 __proto__ 引用。構(gòu)造函數(shù)顯式聲明的 prototype 屬性只是一個指向它實例的原型的引用;實例的原型仍然是通過 __proto__ 引用得到。點此鏈接詳細了解。
你可以在文章 ES3. 7.1 OOP: The general theory 中找到關(guān)于 OPP 通用概念(包括基于類、基于原型等的詳細介紹)的詳細討論。
現(xiàn)在我們已經(jīng)了解了 ECMAScript 對象間的基本關(guān)系,讓我們更深入的了解 JS 運行時系統(tǒng)。我們將會看到,幾乎所有的東西都可以用對象表示。
執(zhí)行上下文為了執(zhí)行 JS 代碼并追蹤其運行時的計算,ECMAScript 規(guī)范定義了執(zhí)行上下文(execution context)的概念。邏輯上執(zhí)行上下文是用棧來保持的(執(zhí)行上下文棧我們一會就會看到),它與調(diào)用棧(call stack)的通用概念相對應(yīng)。
定義7:執(zhí)行上下文:執(zhí)行上下文是一個規(guī)范策略,用于追蹤代碼的運行時計算。
ECMAScript 代碼有幾種類型:全局代碼、函數(shù)和 eval ;它們都在各自的執(zhí)行上下文中運行。不同的代碼類型及其適當(dāng)?shù)膶ο罂赡軙绊憟?zhí)行上下文的結(jié)構(gòu):例如,生成器函數(shù)(generator functions)會將其生成器對象(generator object)保存在上下文中。
我們看一個遞歸函數(shù)調(diào)用:
function recursive(flag) { // Exit condition. if (flag === 2) { return; } // Call recursively. recursive(++flag); } // Go. recursive(0);
當(dāng)一個函數(shù)調(diào)用時,就創(chuàng)建一個新的執(zhí)行上下文并把它壓入棧 - 這時它就成了活躍的執(zhí)行上下文。當(dāng)函數(shù)返回時,其上下文就從棧中推出。
我們將調(diào)用另一個上下文的上下文稱為調(diào)用者(caller)。被調(diào)用的上下文因此就叫做被調(diào)用者(callee)。在上面的例子中,recursive 函數(shù)同時承擔(dān)著上述兩者角色:調(diào)用者和被調(diào)用者 - 當(dāng)遞歸地調(diào)用自身。
定義8:執(zhí)行上下文棧:執(zhí)行上下文棧是一個后進先出的結(jié)構(gòu),它用來維護控制流和執(zhí)行順序。
在上面的例子中,我們對棧有“壓入-推出”的修改:
我們可以看到,全局上下文一直都在棧的底部,它是在執(zhí)行任何其他上下文之前創(chuàng)建的。
你可以在這篇文章中找到更多關(guān)于執(zhí)行上下文的詳細內(nèi)容。
一般情況下,一個上下文中的代碼會運行到結(jié)束,然而正如我們上面所提到的,一些對象 - 如生成器,可能會違反棧后進先出的順序。一個生成器函數(shù)可能會掛起其運行上下文并在完成之前將其從棧中移除。當(dāng)生成器再次激活時,其上下文恢復(fù)并再次被壓入棧:
function *gen() { yield 1; return 2; } let g = gen(); console.log( g.next().value, // 1 g.next().value, // 2 );
上面代碼中的 yield 語句返回值給調(diào)用者并將上下文推出。第二次調(diào)用 next 時,相同的上下文再次被壓入棧并恢復(fù)。這樣的上下文會比創(chuàng)建它的調(diào)用者生命周期更長,因此違反了后進先出的結(jié)構(gòu)。
注:你可以閱讀這篇文檔了解關(guān)于生成器和迭代器的更多內(nèi)容。
現(xiàn)在我們將討論執(zhí)行上下文的重要組成部分;特別是 ECMAScript 運行時如何管理變量的存儲和代碼中嵌套塊創(chuàng)建的作用域(scope)。這是詞法環(huán)境(lexical environments)的通用概念,它在 JS 中用來存儲數(shù)據(jù)和解決“函數(shù)參數(shù)問題(Funarg problem)” - 和閉包(closure)的機制一起。
環(huán)境每個執(zhí)行上下文都有一個相關(guān)的詞法環(huán)境。
定義9:詞法環(huán)境:詞法環(huán)境是用于定義上下文中出現(xiàn)的標(biāo)識符與其值之間的關(guān)聯(lián)的結(jié)構(gòu)。每個環(huán)境都可以有一個指向其可選父環(huán)境的引用。
所以,一個環(huán)境是在某個范圍內(nèi)定義的變量,函數(shù)和類的存儲。
從技術(shù)上來說,一個環(huán)境是由一個環(huán)境記錄(一個將標(biāo)識符映射到值的實際存儲表)和一個對父項(可能是 null)的引用這一對組成。
看代碼:
let x = 10; let y = 20; function foo(z) { let x = 100; return x + y + z; } foo(30); // 150
上面代碼的全局上下文和 foo 函數(shù)的上下文的環(huán)境結(jié)構(gòu)如下圖所示:
從邏輯上講,這使我們想起上面討論過的原型鏈。并且標(biāo)識符解析的規(guī)則也非常相似:如果在自己的環(huán)境中找不到變量,則嘗試在父級環(huán)境中、在父級父級中查找它,以此類推 - 直到整個環(huán)境鏈都查找完成。
定義10:標(biāo)識符解析:在環(huán)境鏈中解析變量(綁定)的過程。 無法解析的綁定會導(dǎo)致 ReferenceError 。
這就解釋了:為什么變量 x 被解析為 100,而不是 10 - 它是直接在 foo 自己的環(huán)境中找到;為什么我們可以訪問參數(shù) z - 它也只是存儲在激活環(huán)境中;也是為什么我們可以訪問變量 y - 它是在父級環(huán)境中找到的。
與原型類似,相同的父級環(huán)境可以由多個子環(huán)境共享:例如,兩個全局函數(shù)共享相同的全局環(huán)境。
注:您可以在這篇文章中獲得有關(guān)詞法環(huán)境的詳細信息。
環(huán)境記錄因類型而異。有對象環(huán)境記錄和聲明式環(huán)境記錄。在聲明式記錄之上還有函數(shù)環(huán)境記錄和模塊環(huán)境記錄。每種類型的記錄都有它的特性。但是,標(biāo)識符解析的通用機制在所有環(huán)境中都是通用的,并且不依賴于記錄的類型。
一個對象環(huán)境記錄的例子就是全局環(huán)境記錄。這種記錄也有相關(guān)聯(lián)的綁定對象,它可以存儲記錄中的一些屬性,而不是全部,反之亦然(譯者注:不同的可以看下面的示例代碼)。綁定對象也可以通過 this 得到。
// Legacy variables using `var`. var x = 10; // Modern variables using `let`. let y = 20; // Both are added to the environment record: console.log( x, // 10 y, // 20 ); // But only `x` is added to the "binding object". // The binding object of the global environment // is the global object, and equals to `this`: console.log( this.x, // 10 this.y, // undefined! ); // Binding object can store a name which is not // added to the environment record, since it"s // not a valid identifier: this["not valid ID"] = 30; **加粗文字** console.log( this["not valid ID"], // 30 );
上述代碼可以表示為下圖:
需要注意的是,綁定對象的存在是為了兼容遺留的結(jié)構(gòu),例如 var 聲明和with 語句,它們也將它們的對象作為綁定對象提供。這就是環(huán)境被表示為簡單對象的歷史原因?,F(xiàn)在,環(huán)境模型更加優(yōu)化,但其結(jié)果是,我們無法再將綁定作為屬性訪問(譯者注:如上面的代碼中我們不能通過 this.y 訪問 y 的值)。
我們已經(jīng)看到環(huán)境是如何通過父鏈接相關(guān)聯(lián)的。現(xiàn)在我們將看到一個環(huán)境的生命周期如何比創(chuàng)造它的上下文環(huán)境的更久。這是我們即將討論的閉包機制的基礎(chǔ)。
閉包ECMAScript中的函數(shù)是頭等的(first-class)。這個概念是函數(shù)式編程的基礎(chǔ),這些方面也被 JavaScript 所支持。
定義11:頭等函數(shù):它是一種函數(shù),其可以作為正常數(shù)據(jù)參與:存儲在變量中,作為參數(shù)傳遞,或作為另一個函數(shù)的值返回。
與頭等函數(shù)概念相關(guān)的是所謂的“函參問題(Funarg problem)”(或“一個函數(shù)參數(shù)的問題”)。當(dāng)一個函數(shù)需要處理自由變量時,這個問題就會出現(xiàn)。
定義12:自由變量:一個既不是參數(shù)也不是自身函數(shù)的局部變量的變量。
我們來看看函參問題,并看它在 ECMAScript 中是如何解決的。
考慮下面的代碼片段:
let x = 10; function foo() { console.log(x); } function bar(funArg) { let x = 20; funArg(); // 10, not 20! } // Pass `foo` as an argument to `bar`. bar(foo);
對于函數(shù) foo 來說,x 是自由變量。當(dāng) foo 函數(shù)被激活時(通過
funArg 參數(shù)) - 應(yīng)該在哪里解析 x 的綁定?是創(chuàng)建函數(shù)的外部作用域還是調(diào)用函數(shù)的調(diào)用者作用域?正如我們所見,調(diào)用者即 bar 函數(shù),也提供了 x 的綁定 - 值為 20 。
上面描述的用例被稱為 downward funarg problem,即在確定綁定的正確環(huán)境時的模糊性:它應(yīng)該是創(chuàng)建時的環(huán)境,還是調(diào)用時的環(huán)境?
這是通過使用靜態(tài)作用域的協(xié)議來解決的,也就是創(chuàng)建時的作用域。
定義13:靜態(tài)作用域:一種實現(xiàn)靜態(tài)作用域的語言,僅僅通過查看源碼就可以確定在哪個環(huán)境中解析綁定。
靜態(tài)作用域有時也被稱為詞法作用域,因此也就是詞法環(huán)境的命名由來。
從技術(shù)上來說,靜態(tài)作用域是通過捕獲創(chuàng)建函數(shù)的環(huán)境來實現(xiàn)的。
注:您可以閱讀鏈接文章的了解靜態(tài)和動態(tài)作用域。
在我們的例子中,foo 函數(shù)捕獲的環(huán)境是全局環(huán)境:
我們可以看到一個環(huán)境引用了一個函數(shù),而這個函數(shù)又回引了環(huán)境。
定義14:閉包:閉包是捕獲定義環(huán)境的函數(shù)。在將來此環(huán)境用于標(biāo)識符解析。注:一個函數(shù)調(diào)用時是在全新的環(huán)境中激活,該環(huán)境存儲局部變量和參數(shù)。激活環(huán)境的父環(huán)境被設(shè)置為函數(shù)的閉包環(huán)境,從而產(chǎn)生詞法作用域語義。
函參問題的第二個子類型被稱為upward funarg problem。它們之間唯一的區(qū)別是捕捉環(huán)境的生命周期比創(chuàng)建它的環(huán)境更長。
我們看例子:
function foo() { let x = 10; // Closure, capturing environment of `foo`. function bar() { return x; } // Upward funarg. return bar; } let x = 20; // Call to `foo` returns `bar` closure. let bar = foo(); bar(); // 10, not 20!
同樣,從技術(shù)上來說,它與捕獲定義環(huán)境的確切機制沒有區(qū)別。只是這種情況下,如果沒有閉包,foo 的激活環(huán)境就會被銷毀。但是我們捕獲了它,所以它不能被釋放,并被保留 - 以支持靜態(tài)作用域語義。
人們對閉包的理解通常是不完整的 - 開發(fā)人員通??紤]閉包僅僅依據(jù) upward funarg problem(實際上是更合理)。但是,正如我們所看到的,downward 和 upward funarg problem 的技術(shù)機制是完全一樣的 - 就是靜態(tài)作用域的機制。
正如我們上面提到的,與原型類似,幾個閉包可以共享相同的父環(huán)境。這允許它們訪問和修改共享數(shù)據(jù):
function createCounter() { let count = 0; return { increment() { count++; return count; }, decrement() { count--; return count; }, }; } let counter = createCounter(); console.log( counter.increment(), // 1 counter.decrement(), // 0 counter.increment(), // 1 );
由于在包含 count 變量的作用域內(nèi)創(chuàng)建了兩個閉包:increment 和 decrement ,所以它們共享這個父作用域。也就是說,捕獲總是“通過引用” 發(fā)生 - 意味著對整個父環(huán)境的引用被存儲。
有些語言可能捕獲的是值,制作捕獲的變量的副本,并且不允許在父作用域中更改它。但是,重復(fù)一遍,在 JS 中,它始終是對父范圍的引用。
注:引擎的實現(xiàn)可能會優(yōu)化這一步,而不會捕獲整個環(huán)境。只捕獲使用的自由變量,但它們?nèi)匀辉诟缸饔糜蛑斜3植蛔兊目勺償?shù)據(jù)。
你可以在鏈接文章中找到有關(guān)閉包和函參問題的詳細討論。
所有的標(biāo)識符都是靜態(tài)的作用域。然而,在 ECMAScript 中有一個值是動態(tài)作用域的。那就是 this 的值。
thisthis 值是一個特殊的對象,它是動態(tài)地、隱式地傳遞給上下文中的代碼。我們可以把它看作是一個隱含的額外參數(shù),我們可以訪問,但是不能修改。
this 值的目的是為多個對象執(zhí)行相同的代碼。
定義15:this:一個隱式的上下文對象,可以從一個執(zhí)行上下文的代碼中訪問 - 以便為多個對象執(zhí)行相同的代碼。
this 主要的用例是基于類的 OOP。一個實例方法(在原型上定義)存在于一個范例中,但在該類的所有實例中共享。
class Point { constructor(x, y) { this._x = x; this._y = y; } getX() { return this._x; } getY() { return this._y; } } let p1 = new Point(1, 2); let p2 = new Point(3, 4); // Can access `getX`, and `getY` from // both instances (they are passed as `this`). console.log( p1.getX(), // 1 p2.getX(), // 3 );
當(dāng) getX 方法被激活時,會創(chuàng)建一個新的環(huán)境來存儲局部變量和參數(shù)。另外,函數(shù)環(huán)境記錄得到傳遞來的 [[ThisValue]] ,它是根據(jù)函數(shù)的調(diào)用方式動態(tài)綁定的。當(dāng)用 p1 調(diào)用時,this 值恰好是 p1 ,第二種情況下是 p2 。
this 的另一個應(yīng)用是泛型接口函數(shù),它可以用在 mixin 或 traits 中。
在下面的例子中,Movable 接口包含泛型函數(shù) move ,它期望這個 mixin 的用戶實現(xiàn) _x 和 _y 屬性:
// Generic Movable interface (mixin). let Movable = { /** * This function is generic, and works with any * object, which provides `_x`, and `_y` properties, * regardless of the class of this object. */ move(x, y) { this._x = x; this._y = y; }, }; let p1 = new Point(1, 2); // Make `p1` movable. Object.assign(p1, Movable); // Can access `move` method. p1.move(100, 200); console.log(p1.getX()); // 100
作為替代方案,mixin 也可以應(yīng)用于原型級別,而不是像上例中每個實例做的那樣。
為了展示 this 值的動態(tài)性,考慮下面例子,我們把這個例子留給讀者來解決:
function foo() { return this; } let bar = { foo, baz() { return this; }, }; // `foo` console.log( foo(), // global or undefined bar.foo(), // bar (bar.foo)(), // bar (bar.foo = bar.foo)(), // global ); // `bar.baz` console.log(bar.baz()); // bar let savedBaz = bar.baz; console.log(savedBaz()); // global
因為只通過查看 foo 函數(shù)的源代碼,我們不能知道它在特定的調(diào)用中 this 的值是什么,所以我們說 this 值是動態(tài)作用域。
注:您可以在這篇文章中得到關(guān)于如何確定 this 值的詳細解釋,以及為什么上面的代碼是那樣的結(jié)果。
箭頭函數(shù)中 this 值比較特殊:其 this 是詞法的(靜態(tài)的),而不是動態(tài)的。即他們的函數(shù)環(huán)境記錄不提供 this 值,它是從父環(huán)境中獲取的。
var x = 10; let foo = { x: 20, // Dynamic `this`. bar() { return this.x; }, // Lexical `this`. baz: () => this.x, qux() { // Lexical this within the invocation. let arrow = () => this.x; return arrow(); }, }; console.log( foo.bar(), // 20, from `foo` foo.baz(), // 10, from global foo.qux(), // 20, from `foo` and arrow );
就像我們所說的,在全局上下文,this 值是全局對象(全局環(huán)境記錄的綁定對象)。以前只有一個全局對象。在當(dāng)前版本的規(guī)范中,可能有多個全局對象,這是代碼領(lǐng)域(code realms)的一部分。我們來討論一下這個結(jié)構(gòu)。
領(lǐng)域在求值之前,所有 ECMAScript 代碼都必須與一個領(lǐng)域相關(guān)聯(lián)。從技術(shù)上來說,一個領(lǐng)域只是為一個上下文提供全局環(huán)境。
定義16:領(lǐng)域:代碼領(lǐng)域是封裝獨立的全局環(huán)境的對象。
當(dāng)一個執(zhí)行上下文被創(chuàng)建時,它與一個特定的代碼領(lǐng)域相關(guān)聯(lián),這個代碼領(lǐng)域為這個上下文提供了全局環(huán)境。該關(guān)聯(lián)在未來將保持不變。
注:瀏覽器環(huán)境中的直接領(lǐng)域是 iframe 元素,正是它提供了一個自定義的全局環(huán)境。在 Node.js 中,它和 vm 模塊的沙箱類似。
規(guī)范的當(dāng)前版本沒有提供顯式創(chuàng)建領(lǐng)域的能力,但是它們可以由實現(xiàn)隱含地創(chuàng)建。不過有一個將這個API暴露給用戶代碼的提案。
從邏輯上來說,堆棧中的每個上下文總是與其領(lǐng)域相關(guān)聯(lián):
現(xiàn)在我們正在接近 ECMAScript 運行時的全貌了。然而,我們?nèi)匀恍枰吹酱a的入口點和初始化過程。這是由 jobs(作業(yè)) 和 job queues(作業(yè)隊列) 機制管理的。
Job有一些操作可以被推遲的,并在執(zhí)行上下文堆棧上有可用點時立即執(zhí)行。
定義17:Job: Job 是一個抽象操作,當(dāng)沒有其他 ECMAScript 計算正在進行時,該操作啟動 ECMAScript 計算。
Job 在 作業(yè)隊列 中排隊,在當(dāng)前的規(guī)范版本中有兩個作業(yè)隊列 ScriptJobs 和 PromiseJobs。
ScriptJobs 隊列中的初始 job 是我們程序的主要入口 - 初始化已加載且求值的腳本:創(chuàng)建一個領(lǐng)域,創(chuàng)建一個全局上下文,并且與這個領(lǐng)域相關(guān)聯(lián),它被推入堆棧,全局代碼被執(zhí)行。
需要注意的是,ScriptJobs 隊列管理著腳本和模塊兩者。
此外,這個上下文可以執(zhí)行其他上下文,或使其他 jobs 到隊列中排隊。一個可以產(chǎn)生排隊的 job 就是 promise。
如果沒有正在運行的執(zhí)行上下文,并且執(zhí)行上下文堆棧為空,則 ECMAScript 實現(xiàn)會從作業(yè)隊列中移除第一個 job,創(chuàng)建執(zhí)行上下文并開始執(zhí)行。
注:作業(yè)隊列通常由被稱為“事件循環(huán)”的抽象來處理。
ECMAScript 標(biāo)準(zhǔn)沒有指定事件循環(huán),而是將其留給實現(xiàn)決定,但是你可以在鏈接頁面找到一個教學(xué)示例。
示例:
// Enqueue a new promise on the PromiseJobs queue. new Promise(resolve => setTimeout(() => resolve(10), 0)) .then(value => console.log(value)); // This log is executed earlier, since it"s still a // running context, and job cannot start executing first console.log(20); // Output: 20, 10
注:你可以在鏈接文檔中閱讀有關(guān) promise 的更多信息。
async 函數(shù)可以等待(await) promise 執(zhí)行,所以它們也使 promise 作業(yè)排隊:
async function later() { return await Promise.resolve(10); } (async () => { let data = await later(); console.log(data); // 10 })(); // Also happens earlier, since async execution // is queued on the PromiseJobs queue. console.log(20); // Output: 20, 10
注:更多 async 函數(shù)內(nèi)容請點擊鏈接。
現(xiàn)在我們已經(jīng)非常接近當(dāng)前 JS 宇宙的最終畫面。馬上我們將看到我們討論的所有組件的主要所有者 - 代理商(Agents)。
AgentECMAScript中的并發(fā)(concurrency)和并行(parallelism)是使用代理模式(Agent pattern)的實現(xiàn)的。代理模式非常接近參與者模式(Actor pattern) - 一個具有消息傳遞風(fēng)格的輕量級進程。
定義18:Agent:代理是封裝執(zhí)行上下文堆棧、作業(yè)隊列集和代碼領(lǐng)域的抽象概念。
依賴代理的實現(xiàn)可以在同一個線程上運行,也可以在多帶帶的線程上運行。瀏覽器環(huán)境中的 Worker 代理是代理概念的一個例子。
代理的狀態(tài)是相互隔離的,可以通過發(fā)送消息進行通信。一些數(shù)據(jù)可以在代理之間共享,例如 SharedArrayBuffer 。代理也可以組合成代理集群。
在下面的例子中,index.html 調(diào)用 agent-smith.js worker ,傳遞共享的內(nèi)存塊:
// In the `index.html`: // Shared data between this agent, and another worker. let sharedHeap = new SharedArrayBuffer(16); // Our view of the data. let heapArray = new Int32Array(sharedHeap); // Create a new agent (worker). let agentSmith = new Worker("agent-smith.js"); agentSmith.onmessage = (message) => { // Agent sends the index of the data it modified. let modifiedIndex = message.data; // Check the data is modified: console.log(heapArray[modifiedIndex]); // 100 }; // Send the shared data to the agent. agentSmith.postMessage(sharedHeap);
worker 的代碼如下:
// agent-smith.js /** * Receive shared array buffer in this worker. */ onmessage = (message) => { // Worker"s view of the shared data. let heapArray = new Int32Array(message.data); let indexToModify = 1; heapArray[indexToModify] = 100; // Send the index as a message back. postMessage(indexToModify); };
你可以在鏈接頁面得到示例的完整代碼。
(需要注意的是,如果你在本地運行這個例子,請在 Firefox 中運行它,因為由于安全原因,Chrome 不允許從本地文件加載 web worker)
下圖展示了 ECMAScript 運行時:
如圖所示,那就是在 ECMAScript 引擎下發(fā)生的事情!
現(xiàn)在文章到了結(jié)尾的時候。這是我們可以在概述文章中涵蓋的 JS 核心的信息量。就像我們提到的,JS 代碼可以被分組成模塊,對象的屬性可以被 Proxy 對象追蹤等等。 - 許多用戶級別的細節(jié)可以在 JavaScript 語言的不同文檔中找到。
盡管我們試圖表示一個 ECMAScript 程序本身的邏輯結(jié)構(gòu),希望能夠澄清這些細節(jié)。如果你有任何問題,建議或反饋意見,我將一如既往地樂于在評論中討論這些問題。
我要感謝 TC-39 的代表和規(guī)范編輯幫助澄清本文。該討論可以在這個 Twitter 主題中找到。
祝學(xué)習(xí) ECMAScript 好運!
Written by: Dmitry Soshnikov
Published on: November 14th, 2017
文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/92432.html
摘要:在線挑戰(zhàn),還沒用過,貌似現(xiàn)在對英文資料心里還有種抵觸,必須克服實驗樓研發(fā)工程師包含了等學(xué)習(xí)課程。書的作者就是開發(fā)了用于數(shù)據(jù)分析的著名開源庫的作者英文資料,對數(shù)據(jù)分析中要用到的一些庫,等等做了簡要介紹。形式的資料,示例代碼都很全。 showImg(https://segmentfault.com/img/remote/1460000004852849); 一、說明 面對網(wǎng)絡(luò)上紛繁復(fù)雜的資...
摘要:一直都挺喜歡這個社區(qū)的,給人的第一感覺就是比較的專業(yè)正式,社區(qū)內(nèi)氛圍不錯,各種文章的質(zhì)量也很好,并且?guī)椭宋液芏唷:荛_心能夠來到這里,記錄自己的成長,希望自己能夠多活躍一下,無論是在問答上面還是寫作上面。 一直都挺喜歡 Segmentfault 這個社區(qū)的,給人的第一感覺就是比較的專業(yè)正式,社區(qū)內(nèi)氛圍不錯,各種文章的質(zhì)量也很好,并且?guī)椭宋液芏?。很開心能夠來到這里,記錄自己的成長,希望...
摘要:這是我收集的一些資源,分享給大家,全部放在百度網(wǎng)盤,有需要的請轉(zhuǎn)存到自己的網(wǎng)盤或者下載,以免網(wǎng)盤鏈接失效,另外還有幾百的視頻文件存在網(wǎng)盤,需要的加全部分享在空間,自己可以去下載與權(quán)威指南配套源碼禪意花園高清源碼基礎(chǔ)教程權(quán)威指南參考手冊鋒利 這是我收集的一些資源,分享給大家,全部放在百度網(wǎng)盤,有需要的請轉(zhuǎn)存到自己的網(wǎng)盤或者下載,以免網(wǎng)盤鏈接失效,另外還有幾百G的視頻文件存在網(wǎng)盤,需要的加...
摘要:軟件的復(fù)雜性命名的藝術(shù)在計算機科學(xué)中只有兩件困難的事情緩存失效和命名規(guī)范。到目前為止,我們依然將看做為開發(fā)人員找不到合適命名的一種替代方式。 軟件的復(fù)雜性:命名的藝術(shù) 在計算機科學(xué)中只有兩件困難的事情:緩存失效和命名規(guī)范?!?Phil Karlton 前言 編寫優(yōu)質(zhì)代碼本身是一件很困難的事情,為什么這么說?因為良好的編碼風(fēng)格是為了能更好的理解與閱讀。通常我們會只注重前者,而忽略了后者...
閱讀 2138·2021-09-27 14:04
閱讀 1883·2019-08-30 15:55
閱讀 1707·2019-08-30 13:13
閱讀 1076·2019-08-30 13:07
閱讀 2754·2019-08-29 15:20
閱讀 3247·2019-08-29 12:42
閱讀 3345·2019-08-28 17:58
閱讀 3606·2019-08-28 17:56