摘要:正大力推進,網景通訊公司即將與他們達成一項協議,讓可以用在瀏覽器上。年月,網景通訊公司和達成協議將被重新命名為,它將會作為瀏覽器中小型客戶端任務的一種腳本語言,同時將會被提升為一種更大的開發富組件的專業工具。
本文轉載自:眾成翻譯
譯者:網絡埋伏紀事
審校: 為之漫筆
鏈接:http://www.zcfy.cc/article/2389
原文:https://auth0.com/blog/a-brief-history-of-javascript/
JavaScript 毋庸置疑是當今最重要的語言之一。Web 的興起已經把 JavaScript 帶到一個前所未有的地步。下面我們來看看 JavaScript 在其短短歷史中是如何演變的,以及它在走向何處。請繼續讀下去!
*
這一切都開始于九十年代一切都發生在 1995 年 5 月到 11 月這六個月內。網景通訊公司在早期的 Web 中擁有強大的地位。它的瀏覽器 Netscape Communicator,作為第一款流行 Web 瀏覽器 NCSA Mosaic 的競爭對手,正獲得廣泛認同。網景是由 90 年代早期參與 Mosaic 開發的同一伙人創立的,而現在,有了錢和自主性,他們就有了尋求進一步擴展 Web 的途徑所需的自由。而這種自由催生了 JavaScript。
網景通訊的創始人及前 Mosaic 團隊的成員 Marc Andreessen 預見到 Web 需要某種方法變得更動態。動畫、交互和其它形式的小動畫應該是未來 Web 的一份子。所以 Web 需要一種能與 DOM 交互(不是跟你現在看到的這樣一成不變)的小腳本語言。不過,這種腳本語言不應該面向大佬開發者,以及在軟件工程方面有經驗的人們——在當時這是一種重要的戰略呼聲。當時 Java 也在興起,并且 Java applets 很快就要成為現實。所以這個用于 Web 的腳本語言需要迎合另一群受眾:設計師。實際上,那時 Web 是靜態的。HTML 依然年輕,并且足夠簡單,非程序員也很容易學得會。所以,要讓 Web 變得更動態,不管是瀏覽器的哪一部分,都應該讓非程序員容易理解。這樣 Mocha 的想法就誕生了。Mocha 要成為用于 Web 的一種腳本語言,它必須是簡單、動態的,并且讓非程序員容易理解。
此時,JavaScript 之父 Brendan Eich走上了歷史舞臺。網景通訊公司雇傭 Eich,是讓他開發一種 “用于瀏覽器的 Scheme”。Scheme 是一種 Lisp 的方言,語法很簡單,它動態而強大,并且本質上是函數式的。而 Web 需要類似這樣的語言:語法容易掌握;動態的,以減少代碼,加快開發;并且強大。Eich 看到有機會可以從事自己喜歡的事情,于是就入伙了。
當時,迫于壓力,必須盡快趕出一個工作原型。當時原名為 Oak 的 Java 語言正開始推動。Sun Microsystems 正大力推進 Java,網景通訊公司即將與他們達成一項協議,讓 Java 可以用在瀏覽器上。那么為什么要開發 Mocha(JavaScript 早期的名字)呢?為什么已經有了 Java,卻還要開發一個全新的語言呢?當時的想法是,Java 不適合 Mocha 的目標受眾:測試腳本編寫人員、業余愛好者、設計師。對于用在瀏覽器這個角色上來說,Java 確實太大太重了。所以當時他們的想法是讓 Java 用于大型專業級組件開發;而 Mocha 將用于小型腳本任務。也就是說,Mocha 命中注定就是 Java 的腳本同伴,在某種程度上類似于 Windows 平臺上 C/C++ 和 Visual Basic 之間的關系。
與此同時,網景的工程師開始詳細地研究 Java。他們甚至開始開發自己的 Java 虛擬機。不過,這個虛擬機很快被否決,因為它從來就沒有實現與 Sun 的虛擬機的完美一致。
此時有很多來自于內部的壓力,要盡可能快地選擇一門語言。Python、Tcl 以及 Scheme 本身都是可能的候選。所以,Eich 必須快搞。不過,他比別人有兩個優勢:可以自由挑選適合的特性集、可以直達拍板的人。不幸的是,他也有一個大的劣勢:沒有時間。必須要做出很多重要的決定,而做出決定的可用時間又很短。JavaScript,即 Mocha,就是在這種背景下誕生的。幾周之內,一個工作原型就推出了,然后就被集成到 Netscape Communicator 中。
于是,本應是用于瀏覽器的 Scheme,現在就大相徑庭了。與 Sun 達成協議的壓力,以及讓 Mocha 變成 Java 的腳本同伴,束縛住了 Eich 的手腳。新語言需要采用類似 Java 的語法,對于很多常用語還采用了熟悉的語義。所以 Mocha 一點也不像 Scheme。它表面上看像是一種動態的 Java,實際上卻是Scheme 和 Self 的早產私生子,但長的像 Java。
1995 年 5 月, Mocha 的原型被集成到 Netscape Communicator 中。很快,它被重命名為 LiveScript。當時,"live" 這個單詞只是為了營銷方便。1995 年 12 月,網景通訊公司和 Sun 達成協議:Mocha/LiveScript 將被重新命名為 JavaScript,它將會作為瀏覽器中小型客戶端任務的一種腳本語言,同時 Java 將會被提升為一種更大的、開發富 Web 組件的專業工具。
第一版的 JavaScript 敲定了該語言中很多現在知名的特性,特別是其對象模型以及函數式特性在此版本中已經出現了。
不同的實現如果當時 Eich 未能按時趕出一個工作原型,很難說會發生什么。其他可選方案一點也不像 Java。Python、Tcl 和 Scheme 都與 Java 大不相同。對于 Sun 公司來說,很難接受一個與 Java 如此不同的同伴語言,或者在歷史和開發上比 Java 本身早的語言。另一方面,Java 很長一段時間是 Web 的一個重要部分。如果 Sun 從沒有過這樣的地位,網景可能會在挑選這樣一個語言上有更多的自由。這是肯定的。不過,如果就算網景自己內部能控制和開發,它會不會選擇采用外部的解決方案呢?我們將永遠不會知道。
當 Sun 和 Netscape 達成協議,將 Mocha/LiveScript 的名稱改為 JavaScript 時,有個大問題被提出來了:其他實現會怎么辦?實際上,盡管 Netscape 很快成為了當時首選的瀏覽器,不過微軟也正在開發 Internet Explorer。從最開始,JavaScript 就帶來了用戶體驗如此大的差異,競爭瀏覽器沒辦法,只能自己也整一套 JavaScript 的實現。此時(并且很長一段時間),Web 標準還不強大。所以微軟實現了自己版本的 JavaScript,叫做 JScript。從名稱中去掉 “Java”,是為了避免潛在的商標問題。不過,JScript 不僅僅是名稱上的不同。它在實現上也略有不同,特別是與某些 DOM 函數有關的實現上有所不同,由此產生的影響一直波及到多年之后的未來。JavaScript 大戰還發生除了名稱和時間表之外的更多方面上,而它的怪癖正是這些大戰打來的創傷。JScript 的第一個版本包含在 1996 年 8 月發布的 IE 3.0 中。
網景的 JavaScript 實現也采用了一個內部名稱。和 Netscape Navigator 2.0 一起發布的版本被稱為 Mocha。在 1996 年秋天,Eich 為了償還匆忙推出它所欠下的技術債,將 Mocha 的大部分重寫為一個更干凈的實現。這個新版本的網景 JavaScript 引擎叫做 SpiderMonkey。SpiderMonkey 現在依然是 Netscape Navigator 的孫子 Firefox 中 JavaScript 引擎的名稱。
有好幾年,JScript 和 SpiderMonkey 是主要的 JavaScript 引擎。二者共同實現的功能(并非總是兼容)會定義接下來幾年中 Web 的樣子。
主要設計特點盡管 JavaScript 是倉促之作,不過有幾個強大的特性在一開始就具備了。這些特性將 JavaScript 定義為一門語言,盡管有各種怪癖,依然讓它能獨樹一幟。
類 JAVA 的語法是使用一門已有的語言,還是發明一門新的語言,這也不是我能決定的。來自高層工程管理人員的強制命令是這門語言必須“看起來像 Java ”。這實際上也就把 Perl、Python、 Tcl 以及 Scheme 這些已有的語言排除掉了。后來,在 1996 年,John Ousterhout 在給 Tk 做宣傳時還感嘆說,Tcl 錯過了這樣一個很好的機會。我并非驕傲,只不過是很高興我選擇 Scheme 式的一等函數以及 Self 式(盡管很怪異)的原型作為主干。至于 Java 的影響,主要是把數據分成基本類型和對象類型兩種(比如字符串和 String 對象),以及引入了Y2K 日期問題,這真是不幸。 - Brendan Eich 的博客:關于流行
盡管讓 JavaScript 語法接近 Java 并非初衷,不過市場力量讓它變成了這樣。退一步想,即使采用與 Java 不同的語法可能會讓實現某些特性更為方便,但是不可否認,采用熟悉的語法更有助于 JavaScript 的普及。
將如下的 Java 示例:
public class Sample { public static void main(String[] args) { System.out.println("Hello world!"); try { final MissileSilo silo = new MissileSilo("silo.weapons.mil"); silo.launchMissile(args[0]); } catch(Exception e) { System.out.println("Unexpected exception: " + e); } } }
與如下(現代) JavaScript 示例做比較:
console.log("Hello world"); try { const silo = new MissileSilo("silo.weapons.mil"); silo.launchMissile(process.argv[0]); } catch(e) { console.log("Unexpected exception" + e); }函數作為一等對象
在 JavaScript 中,函數只是又一個對象類型。它們可以像任何其它元素一樣傳遞,可以被綁定到變量。在稍后版本的 JavaScript 中,函數甚至可以被拋出為異常。這個特性很有可能是在 JavaScript 開發時受到 Scheme 強烈影響的結果。
var myFunction = function() { console.log("hello"); } otherFunction(myFunction); myFunction.property = "1";
通過讓函數變成一等對象,某些函數式編程模式才成為可能。例如,較新版本的 JavaScript 利用了某些函數式模式:
var a = [1, 2, 3]; a.forEach(function(e) { console.log(e); });
這些模式已經被成功用于很多庫,比如 underscore 和 immutable.js。
基于原型的對象模型盡管基于原型的對象模型是通過 JavaScript 得以流行的,不過它卻是在 Self 語言中首次引入。Eich 對這種模型有種強烈的偏好,它足夠強大,能夠模仿像 Java 或 C++ 這種基于 Simula 的語言中的更傳統的方式。實際上,JavaScript 之后的版本中實現的類,也只不過是在原型系統之上的語法糖。
JavaScript 的原型靈感來自于 Self,而 Self 的設計目標之一就是要避免 Simula 風格的對象的問題。特別是,在 Simula 的方式下,類和實例之間的對立被看到是很多固有問題的誘因。有人認為,因為類為對象實例提供某種原型,隨著代碼演變和逐漸變大,就越來越難讓這些基類適應不可預料的新需求。通過將實例作為新對象構建的原型,這種限制就被克服了。因此,原型的概念是:一個通過提供自己的行為,填補新實例的空白的實例。如果一個原型被認為不適合于一個新對象,那么它只需要被克隆和修改,而不會影響所有其它子實例。這在基于類的方式中是挺難做到的(即,修改基類)。
function Vehicle(maxSpeed) { this.maxSpeed = maxSpeed; } Vehicle.prototype.maxSpeed = function() { return this.maxSpeed; } function Car(maxSpeed) { Vehicle.call(this, maxSpeed); } Car.prototype = new Vehicle();
原型的威力讓 JavaScript 變得超級靈活,引發了很多帶有自己對象模型的庫的開發。一個流行的庫 Stampit 就重度使用了原型系統,采用在基于類的傳統方法下不可能的方式,來擴充和操作對象。
原型讓 JavaScript 表面上看起來簡單,但是給庫的作者帶來了自主權。
大怪癖:基礎類型與對象也許在匆忙開發的 JavaScript 中,最大的錯誤之一是某些行為類似的對象有不同的類型。例如,字符串字面量("Hello world")的類型與 String 對象(new String("Hello world"))的類型就是不相同的。這就讓我們有時候不得不采用不必要的、容易混淆的類型檢查。
> typeof "hello world" < "string" > typeof new String("hello world") < "object"
然而,在 JavaScript 歷史中,這只是個開始。它的倉促開發真真切切地導致了一些設計失誤。不過,發明一種用于動態 Web 的語言的優勢不能耽擱,其它的一切只有交給歷史了。
追憶往事:看看 Netscape Navigator 2.0 和 3.0余下的是逆襲的、殘酷的歷史。JS 在客戶端戰勝了 Java,競爭的只有 Flash,而 Flash 支持 JS 的后代 ActionScript - Brendan Eich 的博客:流行
JavaScript 的第一個公開發行版被集成到 1995 年發布的 Netscape Navigator 2.0 中。多虧了虛擬化的奇跡和過時軟件網站,我們現在還可以重現那些時刻!
不幸的是,那時很多 JavaScript 的基礎特性并不能用。匿名函數和原型鏈這兩個最強大的特性就遠遠沒有今天這么完善。不過,這些特性已經是該語言設計的一部分,會在后面幾年中正確實現。需要指出的是,在這個發行版中的 JavaScript 解釋器被認為是處于 alpha 狀態。
Netscape Navigator 2
幸運的是,一年后,1996 年發布的 Netscape Navigator 3.0 已經有很大變化:
Netscape Navigator 3
注意視頻中的錯誤是如何給我們發生什么事情的更多信息。這讓我們推測解釋器以一種特殊的方式對待 prototype 屬性。于是我們嘗試用基礎的 Object 實例來替換對象,這樣我們之后就可以修改該對象。嗯,好了,搞定了!至少在某種程度上。test 函數內的賦值貌似什么都沒做。很顯然,還有很多工作需要去做。盡管如此,這個狀態的 JavaScript 對于很多任務是可用的,并且它已經開始流行了。
像正則表達式、JSON 和異常等特性此時依然不能用。在接下來的幾年中,JavaScript 會迅猛發展。
ECMAScript: JavaScript 標準JavaScript 公開發布后的第一次重大改變是以 ECMA 標準化的形式出現。ECMA 是 1961 年成立的一個行業協會,該協會只從事信息和通訊系統的標準化。
JavaScipt 的標準化工作始于 1996 年 11 月。標準號是 ECMA-262,負責的委員會是 TC-39。這時候,JavaScript 已經是很多頁面的流行元素。這份1996 年的新聞稿 說采用 JavaScript 的頁面數量已達 300,000。
JavaScript 和 Java 是開發 Internet 和 Intranet 應用程序的 Netscape ONE 平臺的基礎技術。自去年引入它們的很短一段時間內,新語言快速被開發者接受。根據 www.hotbot.com 統計,在當今互聯網上有 175,000 個新 Java 小程序和超過 300,000 個使用 JavaScript 的頁面。- Netscape 新聞稿
對于這樣一個年輕的語言來說,標準化是一個重要的步驟,不過依然是一個重大的號召。它將 JavaScript 開放給更廣泛的受眾,并且給其它潛在的實現者在語言進化上的發言權。它還充當了約束其它實現者的用途。那時候,人們擔心微軟或者其它人會偏離默認的實現太遠,從而導致分裂。
由于商標的原因,ECMA 委員會不能用 JavaScript 做名字,而其它名稱也有很多人不喜歡。所以經過幾輪磋商后,決定這個用標準來描述的語言將被叫做 ECMAScript。現在,JavaScript 只是 ECMAScript 的商業名稱。
ECMAScript 1 和 2:標準化之路第一個 ECMAScript 標準是基于 Netscape Navigator 4 發布的 JavaScript 版本,它依然缺失重要的特性,比如正則表達式、JSON、異常,以及內置對象的重要方法。不過,在瀏覽器中它工作得不錯。JavaScript 正開始變得越來越好。1997 年 6 月,版本 1 發布了。
Netscape Navigator 4
注意,視頻中那個簡單的原型和函數測試現在可以正常工作了。在 Netscape 4 中很多工作已經在幕后完成了,而 JavaScript 從受益良多。現在我們的示例基本上可以與任何當代瀏覽器運行的一樣了。這對于 JavaScript 第一次發布成一個標準來說,是一個很好的局面。
標準的第二版 ECMAScript 2 的發布是用來糾正 ECMA 和 JavaScript ISO 標準(ISO/IEC 16262)之間的不一致性的,所以語言沒有做任何改動。這個版本發布于 1998 年 6 月。
此版本的 JavaScript 的一個有趣的怪癖是,在編譯時沒有被捕獲的錯誤(這通常是留作為未確定的)交給解釋器任意決定如何處理。這是因為異常還不是該語言的一部分。
ECMAScript 3:第一次大變動ECMAScript 2 后工作在繼續,對該語言的第一次大變更出現了。這個版本帶來了:
正則表達式
do-while 塊
異常和 try/catch 塊
更多有關字符串和數組的內置函數
格式化數字輸出
in 和 instanceof 運算符
更好的錯誤處理
ECMAScript 3 發布于 1999年 12 月。
這個版本的 ECMAScript 流傳甚廣。它被當時的所有主流瀏覽器所支持,而且多年后一直支持。即使到了今天,有些轉譯器依然可以在產生輸出時,以這個版本的 ECMAScript 為目標。這讓 ECMAScript 3 成為了很多庫的基準目標,即使后來版本的標準發布了也是如此。
即使 JavaScript 越來越流行,它依然主要是一種客戶端語言。不過,它很多新特性讓它離打破這種牢籠更近。
2000年 11 月,Netscape Navigator 6 發布。這個版本是對過去版本的重大修訂,支持 ECMAScript 3。大約在一年半后,Firefox 發布。它是一個基于 Netscape Navigator 代碼庫的精簡版瀏覽器,也支持 ECMAScript 3。這些瀏覽器與 IE 一起,繼續推動 JavaScript 的成長。
AJAX 的誕生AJAX,即異步 JavaScript 和 XML,是一種在 ECMAScript 3 年代誕生的技術。雖然它并非標準的一部分,不過微軟為其 IE5 瀏覽器實現了某些對 JavaScript 的擴展。其中之一就是 XMLHttpRequest 功能(以 XMLHTTP ActiveX 控件的形式)。該功能允許瀏覽器執行對服務器的異步 HTTP 請求,從而允許頁面被即時動態更新。雖然術語 AJAX 直到幾年后才被創造出來,但是這種技術早就到處在用了。
術語 AJAX 是由 Adaptive Path 的聯合創始人 Jesse James Garrett 在這篇標志性博客中創造出來的。
XMLHttpRequest 被證明是非常成功的,多年以后被集成到一個多帶帶的標準中(作為 WHATWG 和 W3C 組織的一部分)。
由實現者給語言帶來一些有趣的東西,并且在瀏覽器中實現,從而促進特性的發展,依然是 JavaScript 和相關的 Web 標準(比如 HTML 和 CSS)繼續發展的方式。不過,那時不同派系之間溝通極少,導致拖延和分裂。平心而論,有了任何感興趣的派系提出建議的程序,今天的 JavaScript 開發顯得更有組織。
玩玩 NETSCAPE NAVIGATOR 6
Netscape Navigator 6
這個版本支持異常,之前版本在試圖訪問 Google 時候遇到的主要缺陷。不可思議的是,即使在今天,試圖在這個版本中訪問 Google,也會一個看得見的工作頁面。相比之下,如果試圖用 Netscape Navigator 訪問 Google,就會被缺乏異常、不完整的渲染以及糟糕的布局弄的焦頭爛額。Web 正在快速發展,即使在當時。
玩玩 INTERNET EXPLORER 5
IE5
IE5 也能渲染當前版本的 Google。不過,眾所周知,在實現某些特性上面,IE 和其它瀏覽器之間有很多分歧。這些分歧禍害了 Web 很多年,也是長期以來 Web 開發者受挫之源,因為他們經常不得不為 IE 用戶實現特例。
實際上,要在 IE5 和 IE6 中訪問 XMLHttpRequest 對象,必須要借助于 ActiveX。其它瀏覽器將其實現為原生對象。
var xhr = new ActiveXObject("Microsoft.XMLHTTP");
毫無疑問,是 IE5 率先將 AJAX 理念變成現實。不過,直到 IE7,微軟才開始遵循標準,并更貼近共識。有些老公司網站依然需要老版本的 IE 才能正確運行。
ECMAScript 3.1 和 4:斗爭之年不幸的是,隨后幾年JavaScript 發展并不順利。從 ECMAScript 4 的工作一開始,委員會中就開始出現了強烈的分歧。有一群人認為 JavaScript 需要一些特性來成為一種更強大的語言,這樣就可以用于大型應用程序開發。這群人提出了很多特性,這些特性涉及面廣,變化大。另一群人認為大型應用程序開發不是 JavaScript 適合的方向。由于缺乏一致意見,加上新提出的某些特性的復雜性,將 ECMAScript 4 的發布變得遙遙無期。
實際上針對 ECMAScript 4 的工作在 1999 年 版本 3 剛出爐的時候,就已經開始了。網景公司內部討論了很多有意義的特性。不過,實現這些特性的興趣已經逐漸減弱,并且 2003 年剛過沒多久,在新版本 ECMAScript 上的工作就停止了。一個臨時報告發布了,有些實現者,比如 Adobe(ActionScript)和微軟(JScript.NET),使用這個報告作為其引擎的基礎。2005 年,在 AJAX 和 XMLHttpRequest 的影響之下,再度激發了新版本 JavaScript 的興趣,TC-39 重啟了工作。幾年過去了,特性集變得越來越大。在 ECMAScript 4 開發的最高峰,有如下這些特性:
類
接口
命名空間
包
可選的類型注解
可選的靜態類型檢查
結構類型
類型定義
多方法(Multimethods)
參數化類型
尾調用
迭代器(Iterator)
生成器(Generator)
內省
類型識別的異常處理器
常量綁定
塊作用域
解構
函數表達式
數組推導式(Array comprehensions)
ECMAScript 4 草案將這個新版本描述為編寫大型應用程序而設計。如果你已熟悉 ECMAScript 6/2015,就會注意到很多來自 ECMAScript 4 的特性被重新引入了。
雖然 ES3 靈活,并且在形式上很強大,但是在開發大型軟件系統實踐中,它的抽象能力經常是無法勝任的。由于在 Web 上采用 Ajax 編程,在應用程序中大量使用 ECMAScript 作為插件以及腳本語言,ECMAScript 程序正變得越來越大,越來越復雜。大型程序的開發可以從靜態類型檢查、名稱隱藏、早綁定以及其它優化手段、直接支持面向對象編程等技術上大大受益,而這些都是 ES3 中所缺乏的。 - ECMAScript 4 草案
一則有趣的歷史片段是如下的 Google Docs spreadsheet,這個文件展示了幾種 JavaScript 引擎的實現狀態,以及涉及其中的派系的討論。
開發 ECMAScript 4 的委員會由 Adobe、Mozilla、Opera(以非官方身份)和微軟組成。Yahoo 在大部分標準和特性已經決定了后,進入了這個委員會。Doug Crockford,一個有影響力的 JavaScript 開發者,就是 Yahoo 為此送進委員會的那個人。他鼓吹他的擔憂,強烈反對很多 ECMAScript 4 提議的修改。他從微軟的代表那里獲得了強烈的支持。Crockford 本人說到:
但是結果微軟的委員也有同樣的擔憂 - 他也認為這門語言正變得太大,失去了控制。在我加入委員會之前,他什么都沒有說,因為他擔心,如果微軟試著阻攔這件事,就會被指責為反競爭行為。根據微軟過去的表現,也許他們有一些不錯的理由對此在意 - 并且很顯然,這些擔憂是有理有據的,因為已經發生過。但是我勸他說,微軟應該做正確的事情,并且以他的聲譽,他決定他應該,也能說服微軟。所以微軟就在 ES4 上改變了立場。 - Douglas Crockford — JavaScript 的現狀和未來
開始是懷疑,后來就變成強勢反對 JavaScript。微軟拒絕接受 ECMAScript 4 的所有部分,并且準備采取各種必要的行動來阻止標準獲得通過(甚至法律訴訟)。幸運的是,委員會中的人設法阻止了法律斗爭。不過,缺乏共識有效地阻止了 ECMAScript 4 推進。
微軟的某些人想在這件事情上采取強硬手段,他們想開始建立書面憑據,開始走申訴程序,想做這些額外的法律程序。我可不想有這種事情。我是不同意 ES4,但是僅限于技術層面,并且我想只限于技術層面;我不想讓它變得比以前更麻煩。我只是想搞清楚什么事情該做,所以我設法溫和一點。但是微軟依然采取了極端立場,說他們拒絕接受 ES4 的任何部分。所以事情就變成了兩極分化,但是我認為兩極分化是因為 ES4 團隊拒絕考慮任何其它觀點的結果。那時委員會沒有達成共識,這是件糟糕的事情,因為標準小組必須要達成共識。一個標準不應該是有爭議的。 - Douglas Crockford — JavaScript 的現狀和未來
Crockford 想出一個點子來推進,就是重新弄一個標準,這個標準更簡單,減少一些特性集,這樣所有人都可以同意:沒有新語法,只有來自使用該語言的經歷中的實際提升。這個提案后來被稱為 ECMAScript 3.1。
有一段時間,兩種標準并存,并且設置了兩個非正式的委員會。不過,ECMAScript 4 太復雜,沒辦法在面對沖突的情況完成。ECMAScript 3.1 更簡單,并且盡管在 ECMA 中有斗爭,它還是完成了。
ECMAScript 4 的結束出現在 2008 年,Eich 通過一封電子郵件,發送了一次奧斯陸會議的內容提要,詳細描述了 ECMAScrpt 走向和版本 3.1 和 4 的未來。
這次會議的結論是:
與所有各方充分合作,集中精力完成 ES 3.1,到明年初確定兩個執行標準。
下一步上的合作超出 ES3.1,會包含語法上的擴展,但是會在語義和語法創新上比 ES4 更謹慎。
有些 ES4 提案已經被認為對 Web 不合理,最好不予討論:包、命名空間和早綁定。這個結論對于 Harmony 來說很關鍵。
ES4 的其它目標和理念正被改寫,以保持在委員會中的一致;包括類的概念是基于已有的 ES3 概念結合提議的 ES3.1 擴展。
總之,ECMAScript 4 花了近 8 年的時間開發,最后卻被廢棄了。這對涉及的所有人來說都是一個沉重的教訓。
單詞 "Harmony(和諧)" 出現在上面的結論中。這是將來對 JavaScript 擴展時項目的標準名稱。Harmony 會是所有人都同意的方案。在 ECMAScript 3.1 發布后(以版本 5 的形式,下面我們會看到),所有 JavaScript 中要討論的新主意都會出現在ECMAScript Harmony 中。
ActionScriptActionScript 是一個基于 ECMAScript 4 早期草案的編程語言。Adobe 將其實現為 Flash 應用程序套件的一部分,也是它支持的唯一的腳本語言。這就讓 Adobe 采用強硬的立場來支持 ECMAScript 4,甚至還將他們的引擎(Tamarin)開源發布了,以希望加快 ECMAScript 4 的采納。Adobe 員工 Mike Chambers 爆光了一個在此事上的有趣看法:
ActionScript 3 沒有消失,我們基于最近的決定,沒有從中刪除任何東西..我們會繼續跟蹤 ECMAScript 規范,但是正如我們一直所做的那樣,我們會創新,盡可能推動 Web 向前發展(正如我們在過去已做過的那樣)- Mike Chamber 的博客
ActionScript 開發者期望 ActionScript 中的創新會驅動 ECMAScript 中的特性。不幸的是這事從來沒有出現過,而且后來出現在 ECMAScript 2015 中的特性與 ActionScript 在很多方面不兼容。
有人看到這是微軟嘗試保持控制 ECMAScript 語言和實現的一種策略。此時,唯一可行的 ECMAScript 4 引擎是 Tamarin,所以此時占有 80% 瀏覽器市場份額的微軟可以繼續使用它自己的引擎(以及擴展),而不用承擔切換到競爭對手的替代品的代價或者花時間內部實現一切。 其他人只是說微軟的異議僅僅是技術上的,跟來自 Yahoo 的人一樣。Microsoft 的引擎 JScript 此時與其它實現有很多差異。有人已經看到這是保持秘密控制該語言的一種手段。
ActionScript 目前依然是 Flash 的開發語言,而 Flash 隨著 HTML5 的到來,逐漸淡出了人們的視野。
如果 ECMAScript 4 已經被流行 JavaScript 引擎實現的話,ActionScript 依然與它最像:
package { import flash.display.Sprite; public class MyRectangle_v3 extends Sprite { private var _outlineWeight:Number; private var _color:uint; private var _xLocation:int; private var _yLocation:int; private var _rectangleWidth:int; private var _rectangleHeight:int; public function MyRectangle_v3(outlineWeight:Number, color:uint, xLocation:int, yLocation:int, rectangleWidth:int, rectangleHeight:int) { _outlineWeight = outlineWeight; _color = color; _xLocation = xLocation; _yLocation = yLocation; _rectangleWidth = rectangleWidth; _rectangleHeight = rectangleHeight; } public function draw():void{ graphics.lineStyle(_outlineWeight); graphics.beginFill(_color); graphics.drawRect(_xLocation, _yLocation, _rectangleWidth, _rectangleHeight); graphics.endFill(); } } }E4X?什么是 E4X?
E4X 是一個公認的 ECMAScript 擴展的名稱。它在 ECMAScript 4 開發期間發布(2004年),所以就采用了綽號 E4X。其實際名稱是 ECMAScript for XML,并被標準化為 ECMA-357。E4X 擴充了 ECMAScript,以支持對 XML 內容的原生處理和解析。在 E4X 中,XML 被當作是一種原生數據類型。它最初被主流 JavaScript 引擎(比如 SpiderMonkey)采納,不過之后由于很少有人用而被拿掉。在 Firefox 版本 21 中被刪除。
除了其名稱中有數字 "4" 之外,E4X 與 ECMAScript 4 沒多大關系。
如下是一個使用 E4X 的示例:
var sales =; alert( sales.item.(@type == "carrot").@quantity ); alert( sales.@vendor ); for each( var price in sales..@price ) { alert( price ); } delete sales.item[0]; sales.item += - ; sales.item.(@type == "oranges").@quantity = 4;
可以說,其它數據格式(比如 JSON)已經在 JavaScript 社區中獲得了更廣泛的認同,所以 E4X 出現和消失都沒惹啥亂子。
ECMAScript 5:JavaScript 的重生在 ECMAScript 4 的漫長斗爭之后,從 2008 年開始,社區就在注意力放在 ECMAScript 3.1 上。ECMAScript 4 被廢棄。在 2009 年,ECMAScript 3.1 完成,并且涉及的各方都簽字確認了。而 ECMAScript 4 即使還沒有真正發布,也已經被公認為是 ECMAScript 的一個特定變種,所以委員會決定將 ECMAScript 3.1 重新命名為 ECMAScript 5,以避免混淆。
ECMAScript 5 成為了最受支持的 JavaScript 版本之一,也成為了很多轉譯器的編譯目標。ECMAScript 5 被 Firefox 4 (2011)、Chrome 19 (2012)、Safari 6 (2012)、Opera 12.10 (2012) 和 Internet Explorer 10 (2012)完全支持。
ECMAScript 5 是對 ECMAScript 3 的一種相當謹慎的更新,它包括:
Getter/setters
數組和對象字面量中的尾隨逗號
保留關鍵字可以作為屬性名
新的 Object 方法(create、defineProperty、keys、seal、freeze、getOwnPropertyNames 等等)
新的 Array 方法(isArray、indexOf、every、 some、map、filter、reduce 等等)
String.prototype.trim 和屬性訪問
新的 Date 方法(toISOString、now、toJSON)
函數 bind
JSON
不可變的全局對象(undefined、NaN、Infinity)
嚴格模式
其它次要的變更(parseInt 忽略前導零、拋出的函數有正確的 this 值,等等)
所有更新都不需要語法上的修改。那時 getter 和 setter 已經被很多瀏覽器非正式地支持。新的 Object 方法通過給程序員更多工具來確保強制某些不變性,來改進“大型程序的編寫”(Object.seal、Object.freeze、Object.createProperty)。嚴格模式通過阻止很多常見的錯誤源,在這一領域也成為一種強大的工具。額外的 Array 方法改進了某些函數式范式(map、reduce、filter、every、some)。另一個大變化是 JSON:一個受 JavaScript 啟發的數據格式,現在通過 JSON.stringify 和 JSON.parse 原生支持了。其它變化基于實踐經驗在幾個方面作出了小的改進。總而言之,ECMAScript 5 是適度的改進,幫助 JavaScrpt 成為一種對于小腳本和較大的項目來說都更可用的語言。依然有很多來自 ECMAScript 4 的好點子被廢棄,并且你會看到這些好點子又通過 ECMAScript Harmony 提案回歸。
2011 年,ECMAScript 5 以 ECMAScript 5.1 的形式又來了一次迭代。這個版本澄清了標準中一些容易引起歧義之處,但是沒有提供任何新特性。所有的新特性定于在下一個 ECMAScript 的大發布中。
ECMAScript 6 (2015) 和 7 (2016): 一個通用的語言ECMAScript Harmony 提案成為將來對 JavaScript 的改進的中心。ECMAScript 4 的很多想法被永久封存了,但另外一些則又以新的面目被重新啟用。ECMAScript 6,后來被重新命名為 ECMAScript 2015,被指定為帶來大變化。幾乎所有需要在語法上改變的更新都被放到這個版本中。不過,這次委員會達成了一致,ECMAScript 6 最終于 2015 年發布。很多瀏覽器廠家已經開始著手實現它的特性,但是大的變動需要點時間。直到今天,并非所有瀏覽器都完全覆蓋 ECMAScript 2015(雖然它們已經很接近)。
ECMAScript 2015 的發布導致轉譯器的使用大幅度增加,比如 Babel 或 Traceur。甚至在它發布之前,因為這些轉譯器跟蹤了技術委員會的進展,人們已經體驗了很多 ECMAScript 2015 的好處。
一些 ECMAScript 4 的大特性在這個版本的 ECMAScript 中被實現。不過,實現的出發點不一樣了。例如,ECMAScript 2015 中的類只不過是在原型之上的語法糖。這種思維模式減輕了新特性的過渡和開發。
在我們的JavaScript 2015 特性概述一文中,我們對 ECMAScript 2015 的新特性做了一個全面的概述。你也可以看看 ECMAScript 兼容性表,了解一下在實現方面我們現在的確切位置。
新特性小結如下:
Let(詞法上的)和 const(不可重新綁定的)綁定
箭頭函數(匿名函數的簡寫)以及詞法 this(包含作用域 this)
類(原型之上的語法糖)
對象字面量提升(計算鍵、短方法定義等等)
模板字符串
Promise
Generator、iterable、iterator 和 for..of
函數的默認參數及剩余運算符
擴展語法
解構
模塊語法
新集合(Set、Map、WeakSet、WeakMap)
代理和反射
Symbol 數據類型
類型化數組
內置支持子類化
有保證的尾調用優化
更簡單的 Unicode 支持
二進制和八進制字面量
類、let、const、promise、generator、iterators、模塊,等等。這些特性都是為了把 JavaScript 帶給更多受眾,幫助開發大型應用程序。
你也許會驚訝,在 ECMAScript 4 失敗之時,還有這么多特性能闖過標準化過程這一關。從這個層面上,必須得指出,ECMAScript 4 中很多最具侵入性的特性都沒有被重新考慮,比如,命名空間、可選類型;同時,其它的特性被以可以通過之前的異議的方式重新考慮了,比如,讓類成為原型之上的語法糖。ECMAScript 2015 依然是個苦差事,花了約 6 年完成(并且需要更長時間完全實現)。不過,這樣艱巨的一個任務能被 ECMAScript 技術委員會完成,也可以看作是個好兆頭。
2016 年,一個 ECMAScript 的小修訂版發布了。這個小修訂版是 TC-39 實施的新發布過程的結果。所有新提案必須經過四個階段的過程。每個達到第四個階段的提案有很大機會會被包含在下一個版本的 ECMAScript 中(不過委員會依然可以選擇推遲將其列入議程)。這樣,提案就幾乎可以獨立開發(不過與其它提案的交互必須在考慮之列)。提案不會中斷 ECMAScript 的開發。如果一個提案已經準備列入,并且足夠的提案已經達到第四階段,那么就可以發布一個 ECMAScript 新版本。
2016 年發布的版本是一個相當小的版本。它包括:
取冪運算符(**)
Array.prototype.includes
一些小的更正(generator 不能與 new 一起用等等)。
不過,在 2016 年,某些有趣的提案已經達到第四階段,所以 ECMAScript 的未來是什么呢?
未來與超越:ECMAScipt 2017 及以后也許目前進行中的最重要的第四階段提案是 async/await。Async/await 是對 JavaScript 的一種語法擴展,可以讓處理 promise 變得更爽。例如,對于如下 ECMAScript 2015 代碼:
function apiDoSomethingMoreComplex(withThis) { const urlA = "..."; const urlB = "..."; httpLib.request(urlA, withThis).then(result => { const parsed = parseResult(result); return new Promise((resolve, reject) => { database.update(updateStatement, parsed).then(() => { resolve(parsed); }, error => { reject(error); }); }); }).then(result => { return httpLib.request(urlB, result); }).then(result => { return worker.processData(result); }).then(result => { logger.info(`apiDoSomethingMoreComplex success (${result})`); }, error => { logger.error(error); }); }
把它與如下的 async/await 代碼比較:
async function apiDoSomethingMoreComplex(withThis) { const urlA = "..."; const urlB = "..."; try { let result = await httpLib.request(urlA, withThis); const parsed = parseResult(result); await database.update(updateStatement, parsed); result = await httpLib.request(urlB, parsed); result = await worker.processData(result); logger.info(`apiDoSomethingMoreComplex success (${result})`); } catch(e) { logger.error(e); } }
其它第四階段的提案在范圍上都是次要的:
Object.values 和 Object.entries
字符串補白
Object.getOwnPropertyDescriptors
函數參數允許尾隨逗號
這些提案都是定于在 2017 年發布,不過委員會可能會選擇自行決定推遲。不過,僅是有async/await 這一點就是一件讓人激動的變動。
但是未來并非止步于此!我們可以看看其它的一些提案,了解一下更遠的未來會出現什么。一些有趣的提案是:
SIMD API
異步迭代(async/await + 迭代)
Generator 箭頭函數
64 位 整型操作
Realm(狀態分離/隔離)
共享內存和原子
JavaScript 正越來越像一門通用的語言。不過,還有一件對 JavaScript 的未來會產生重大影響的大事情。
WebAssembly如果你還沒聽說過 WebAssembly,就應該讀讀這篇文章。自 ECMAScript 5 發布以來,引發的庫、框架和一般開發的激增,已經讓 JavaScript 變成了對其它語言的一個有興趣的目標。對于大的代碼庫,可互操作性是關鍵。比如說游戲。游戲開發的通用語言依然是 C++,并且它對很多架構來說都是可移植的。將一個 Windows 游戲或者電子游戲移植到瀏覽器上被看作是一件不可能實現的任務。不過,當前 JIT JavaScript 虛擬機不可思議的性能讓這成為可能。于是,像 Emscripten 這種 LLVM-to-JavaScript 編譯器應運而生。
Mozilla 看到了這點,并開始著手研究讓 JavaScript 變成編譯器的合適目標。Asm.js 誕生了。Asm.js 是 JavaScript 的一個嚴格子集,用來作為編譯器的目標。JavaScript 虛擬機可以被優化為識別這個子集,并且生成比目前可能在普通 JavaScript 代碼中更好的代碼。瀏覽器慢慢變成了一個編譯應用的全新目標,而 JavaScript 在其中心。
不過,有些限制是 Asm.js 也不能解決了。畢竟這與 JavaScript 的用途無關,所以必須要對 JavaScript 作出改變。要讓 web 成為其它語言的合適目標,就需要點不同的東西,而這正是 WebAssembly 所有的。WebAssembly 是用于 Web 的字節碼。任何帶有合適編譯器的程序,都可以被編譯為 WebAssembly,然后運行在合適的虛擬機上(JavaScript 虛擬機可以提供所需的語義)。實際上,首版 WebAssembly 就以與 Asm.js 規范一對一兼容為目標。WebAssembly 不僅帶來了加載時間更快的承諾(解析字節碼比解析文本更快),還帶來了 Asm.js 中目前還不能用的可能優化。想像一下,JavaScript 和你已有的代碼之間能完美互用的 Web。
乍一看,這好像是危及到 JavaScript 的發展,然而事實恰好相反。通過讓其它語言和框架更容易與 JavaScript 互用,JavaScript 就可以繼續發展為一門通用的語言。而WebAssembly 是為此必需的工具。
目前,Chrome、Firefox 和 Microsot Edge 的開發版支持 WebAsembly 規范草案,并且能夠運行演示應用程序。
總結JavaScript 的歷史漫長而崎嶇。它先被提議為用于 Web 的 Scheme,然后早早就被類 Java 的語法束縛住了。它的第一個原型在幾周之內就被開發出來了。受市場之害,它在兩年之內變了三個名字。然后被標準化了,同時得到一個聽起來像皮膚病的名字。在三次成功的發布后,第四版陷入開發地獄約八年。開發方向一變再變,莫衷一是。然后,純通過一個特性(AJAX)的成功,社區又重歸行動一致,恢復了開發。第 4 版被廢棄,一個次要版本,即人人皆知的 第 3.1 版本,被重命名為 第 5 版。第 6 版又在開發上花了很多年,不過這次委員會成功了,雖然又決定改名,但是這次是改為 2015。這個修訂版很大,花了很多時間實現,但是最終,給 JavaScript 帶來了新風。社區像以前一樣活躍。Node.js、V8 和其它項目已經把 JavaScript 帶到了一個前所未有的地步。Asm.js、WebAssembly 正進一步改進它。并且不同階段活躍的提案都讓 JavaScript 的未來像以前一樣光明。這是一條漫長的路,充滿崎嶇,而 JavaScript 依然始終是最成功的語言之一。這本身就是一種證明。永遠把賭注押在 JavaScript 上。
文章版權歸作者所有,未經允許請勿轉載,若此文章存在違規行為,您可以聯系管理員刪除。
轉載請注明本文地址:http://specialneedsforspecialkids.com/yun/84830.html
摘要:一經發布,關注度就居高不下,這也引起了當時的巨頭微軟的注意。微軟為了搶占瀏覽器的市場份額,在年月便在其發布的中搭載了其對的實現。 JavaScript誕生于1995年,最初的想法就是在人們提交表單的時候,提前做一些驗證,提高用戶體驗。試想一下,再那個年代,因為網速很慢,提交個表單,可能要等待20幾秒,再等待這么長的時間之后,服務器返回消息說有一個必填字段沒有填......,介于此,Ne...
摘要:在日益流行期間,一經發布就取得了巨大的成功,于此同時微軟決定在瀏覽器中投入更多的資源,然后微軟就在其瀏覽器中加入了名為。微軟推出其這就意味著有了兩個不同版本的,與其它編程語言不同,當時還沒有標準的規定。 「 JavaScript 」 誕生于 1995年,當時主要是用于處理由服務器負責的一些輸入驗證操作。 在 「 JavaScript 」為誕生之前,必須要把表單數據發送到服務端,然后等到...
摘要:模塊演化簡史從屬于筆者的開發基礎與工程實踐。首個采樣該設計模式的界面庫當屬,其是創建于年。允許我們定義模塊,并且在顯式地聲明其依賴模塊而由框架完成自動注入。而最后的模塊化規范定義于年正式發布,也就是被命名為。 JavaScript 模塊演化簡史 從屬于筆者的 Web 開發基礎與工程實踐。本文主要總結自 The Evolution of JavaScript Modularity、Nat...
摘要:如何看待人工智能的本質人工智能的飛速發展又經歷了哪些歷程本文就從技術角度為大家介紹人工智能領域經常提到的幾大概念與發展簡史。一人工智能相關概念人工智能就是讓機器像人一樣的智能會思考是機器學習深度學習在實踐中的應用。 作為近幾年的一大熱詞,人工智能一直是科技圈不可忽視的一大風口。隨著智能硬件的迭代,智能家居產品逐步走進千家萬戶,語音識別、圖像識別等AI相關技術也經歷了階梯式發展。如何看待...
閱讀 1678·2021-09-26 10:00
閱讀 2938·2021-09-06 15:00
閱讀 3542·2021-09-04 16:40
閱讀 2305·2019-08-30 15:44
閱讀 720·2019-08-30 10:59
閱讀 1888·2019-08-29 18:34
閱讀 3623·2019-08-29 15:42
閱讀 2298·2019-08-29 15:36