JavaScript(通常缩写为JS)是一门基于原型头等函数的多范式高级解释型编程语言[9][10],它支持面向对象程式设计、指令式编程函数式编程。它提供方法来操控文本、数组、日期以及正则表达式等。不支持I/O,比如网络、存储和图形等,但这些都可以由它的宿主环境提供支持。它由Ecma通过ECMAScript实现语言的标准化[9]。目前,它被世界上的绝大多数网站所使用,也被世界主流浏览器ChromeFirefoxSafariOpera)所支持。

事实速览 编程范型, 设计者 ...
JavaScript
Thumb
JavaScript原始码截图
编程范型事件驱动函数式指令式
设计者创造者布兰登·艾克ECMAScript规范的其他关键贡献者
发行时间1995年12月4日,​29年前​(1995-12-04[1]
当前版本
  • ECMAScript 2024(2024年6月;稳定版本)[2]
  • ECMAScript 2025(2024年3月27日;预发布版本)[3]
编辑维基数据链接
类型系统动态类型鸭子类型
文件扩展名
  • .js
  • .cjs
  • .mjs[4]
网站www.ecma-international.org/publications-and-standards/standards/ecma-262/ 编辑维基数据链接
主要实现产品
V8JavaScriptCoreSpiderMonkeyChakra
启发语言
AWK[5]CHyperTalk英语HyperTalkJava[6]LuaPerlPythonSchemeSelf
影响语言
ActionScriptAtScript英语AtScriptCoffeeScriptDartJScript .NETLiveScript英语LiveScriptObjective-J英语Objective-JOpa英语Opa_(programming_language)QMLRakuTypeScript
关闭
事实速览 扩展名, 互联网媒体类型 ...
JavaScript
Thumb
扩展名
.js
互联网媒体类型
application/javascript
text/javascript (obsolete)[7]
统一类型标识com.netscape.javascript-source[8]
格式类型脚本语言
关闭

JavaScript与Java在名字和语法上都很相似,但这两门编程语言从设计之初就有很大不同。JavaScript在语言设计上主要受到了Self(一种基于原型的编程语言)和Scheme(一门函数式编程语言)的影响[10],在语法结构上它和C语言很相似(如if条件语句、switch语句、while循环和do-while循环等)[11]

对于客户端来说,JavaScript通常被实现为一门解释语言,但如今它已经可以被即时编译(JIT)。随着HTML5CSS3语言标准的推行,它还可以用于游戏、桌面和流动应用程序的开发,以及在伺服器端网络环境运行(如Node.js)。

faviconfaviconfaviconfaviconfavicon
11 sources

历史

faviconfaviconfaviconfaviconfavicon
17 sources

肇始于网景

1993年,国家超级计算机应用中心(NCSA)发表了NCSA Mosaic,这是最早流行的图形接口网页浏览器,它在万维网的普及上发挥了重要作用[12]。1994年,Mosaic的主要开发人员创立了Netscape公司,并雇用了许多原来的NCSA Mosaic开发者用来开发Netscape Navigator,该公司的目标是取代NCSA Mosaic成为世界第一的网页浏览器。在四个月内,已经占据了四分之三的浏览器市场,并成为1990年代互联网的主要浏览器[13]

在网络发展的这些年,网页只能是静态的,缺乏在浏览器中加载网页后的动态行为能力。公司的创始人马克·安德森认为HTML需要一种胶水语言,让网页设计师和兼职程序员可以很容易地使用它来组装图片和插件之类的组件,且代码可以直接编写在网页标记中。1995年,网景招募了布兰登·艾克,目标是把Scheme语言嵌入到Netscape Navigator浏览器中[14]。但更早之前,网景已经跟昇阳合作,计划在Netscape Navigator中嵌入Java语言,这时网景内部产生激烈的争论[15]

网景公司管理层很快决定,最佳的方案是由艾克设计一种新的语言,其语法类似于Java,而不像Scheme或其他现存的脚本语言。为了在其他竞争提案中捍卫JavaScript这个想法,公司需要有一个可以运作的原型。艾克在1995年5月仅花了十天时间就把原型设计出来了[16][17]。最初命名为Mocha,1995年9月在Netscape Navigator 2.0的Beta版中改名为LiveScript,同年12月在Netscape Navigator 2.0 Beta 3中部署时又被重命名为JavaScript[1][18],这是因为当时网景公司与昇阳电脑公司组成的开发联盟为了让这门语言搭上Java这个编程语言“热词”,所以才将其临时改名为JavaScript,这成了日后这成为大众对这门语言有诸多误解的原因之一[19]。同年昇阳电脑公司在1995年12月1日正式申请注册JavaScript的商标,于2000年7月28日获批 [20],但此后昇阳电脑公司和之后收购它的甲骨文公司从未使用该商标制造过产品,给JavaScript社区带来了不确定性,阻碍了JavaScript社区的自由发展,直到几十年后,1.5万多名JavaScript社区成员于2024年11月22日联名向美国专利商标局申请终止该商标 [21][22]

faviconfaviconfaviconfaviconfavicon
9 sources

微软采纳

微软公司于1995年首次推出Internet Explorer,引发了与Netscape的浏览器大战。微软对Netscape Navigator解释器进行了逆向工程,创建了JScript,以与处于市场领导地位的网景产品同台竞争。JScript也是一种JavaScript实现,这两个JavaScript语言版本在浏览器端共存意味着语言标准化的缺失。发展初期,JavaScript的标准并未确定,同期就有网景的JavaScript和微软的JScript。除此之外,微软也在网页技术上加入了不少专属物件,使不少网页使用非微软平台及浏览器无法正常显示[23][24]。这导致在浏览器大战期间网页设计者通常会把“用Netscape可达到最佳效果”或“用IE可达到最佳效果”的标志放在主页[23][25]

faviconfaviconfavicon
3 sources

标准化

1996年11月,网景正式向ECMA(欧洲电脑制造商协会)提交语言标准。1997年6月,ECMA以JavaScript语言为基础制定了ECMAScript标准规范ECMA-262。JavaScript成为了ECMAScript最著名的实现之一[26]。除此之外,ActionScriptJScript也都是ECMAScript规范的实现语言。尽管JavaScript作为给非程序人员的脚本语言,而非作为给程序人员的编程语言来推广和宣传,但是JavaScript具有非常丰富的特性。

favicon
1 sources

增长和标准化

在21世纪初Internet Explorer占主导地位期间,客户端脚本停滞不前。这在2004年开始改变,当时Netscape的继任者Mozilla发布了Firefox浏览器。Firefox受到许多人的好评,从Internet Explorer获得了巨大的市场份额。[27]

2005年,Mozilla加入了ECMA International,并开始研究ECMAScript for XML(E4X)标准。这导致Mozilla与Macromedia(后来被Adobe Systems收购)合作,他们正在用基于ECMAScript 4草案的ActionScript 3语言实现E4X。目标是将ActionScript 3标准化为新的ECMAScript 4。为此,Adobe Systems将Tamarin实现作为开源项目发布。然而,Tamarin和ActionScript 3与既定的客户端脚本太不同,如果没有微软的合作,ECMAScript 4从未获取成果。

与此同时,与ECMA工作无关的开源社区正在发生非常重要的发展。2005年,Jesse James Garrett发布了一份白皮书,其中他创造了Ajax一词,并描述了一套技术,其中JavaScript是骨干,用于创建可以在后台加载数据的Web应用程式,避免了重新加载整页的需要。这引发了JavaScript的复兴时期,由开源库和围绕它们形成的社区带头。创建了许多新库,包括jQuery、Prototype、Dojo Toolkit和MooTools。

谷歌于2008年首次推出Chrome浏览器,其V8 JavaScript引擎比竞争对手更快。[28]关键的创新是及时编译(JIT)[29],因此其他浏览器供应商需要为JIT彻底改革他们的引擎[30]

2008年7月,这些不同的政党聚集在一起,在奥斯陆举行会议。这导致在2009年初达成了最终协议,将所有相关工作结合起来,推动语言向前发展。结果是2009年12月发布的ECMAScript 5标准。

faviconfaviconfaviconfavicon
4 sources

走向成熟

关于该语言的雄心勃勃的工作持续了数年,最终随着 2015 年ECMAScript 6的发布而正式形成了广泛的补充和改进。

Ryan Dahl在 2009 年创建的Node.js引发了 Web 浏览器之外 JavaScript 使用的显着增加。Node 结合了V8引擎、事件循环和I/O API,从而提供了独立的 JavaScript 运行时系统。截至 2018 年,Node 已被数百万开发人员使用,并且npm拥有世界上所有包管理器中最多的模块。

ECMAScript 草案规范目前在GitHub上公开维护,并通过定期的年度快照生成版本。对语言的潜在修订通过全面的提案流程进行审查。现在,开发人员不再单独检查即将推出的功能的状态,而不是版本号。

当前的 JavaScript 生态系统拥有许多库和框架、已建立的编程实践以及在 Web 浏览器之外大量使用 JavaScript。另外,随着单页应用程式和其他大量使用 JavaScript 的网站的兴起,已经创建了多个转译器来帮助开发过程。

概论

一般来说,完整的JavaScript包括以下几个部分:

  • ECMAScript,描述了该语言的语法和基本对象;
  • 文档对象模型(DOM),描述处理网页内容的方法和接口;
  • 浏览器对象模型(BOM),描述与浏览器进行交互的方法和接口。

JavaScript的基本特点如下:

  • 是一种解释性脚本语言(代码不进行预编译);
  • 主要用来向HTML页面添加交互行为;
  • 可以直接嵌入HTML页面,但写成单独的js文件有利于结构和行为的分离。

JavaScript常用来完成以下任务:

  • 嵌入动态文本于HTML页面;
  • 对浏览器事件作出响应;
  • 读写HTML元素;
  • 在数据被提交到伺服器之前验证数据;
  • 检测访客的浏览器资讯;
  • 控制Cookie,包括创建和修改等;

特性

不同于伺服器端脚本语言(如PHPASP),JavaScript主要被作为客户端脚本语言在用户的浏览器上运行,不需要伺服器的支持。所以在早期程序员比较青睐于JavaScript以减少对伺服器的负担,而与此同时在安全性上出现了问题。随着伺服器变得强大,现在的程序员更喜欢运行于服务端的脚本以保证安全,但JavaScript仍然以其跨平台、容易上手等优势大行其道。同时,有些特殊功能(如AJAX)必须依赖JavaScript在客户端提供支持。随着引擎(如V8)和框架(如Node.js)的发展,以及事件驱动异步IO等特性,JavaScript也被逐渐用来编写伺服器端程序。

以下是ECMAScript通常所实现的特性。

faviconfaviconfaviconfaviconfavicon
25 sources

指令式与结构化

JavaScript支持许多C语言的结构化编程语法(如if条件语句、while循环、switch语句和do-while循环等),但作用域是一个例外。JavaScript在过去只支持使用var关键字来定义变量的函数作用域,但ECMAScript 2015中加入了let关键字来支持块级作用域[31]。这意味着JavaScript现在既支持函数作用域又支持块级作用域。JavaScript还支持自动在语句末添加分号,允许忽略语句末尾的分号。[32]

faviconfavicon
2 sources

弱类型

JavaScript是弱类型的,这意味着变量可以被隐式转换为另一个类型。[33]

  • 二元运算符+会把两个操作数转换为字符串,除非两个操作数都为数字类型。[34]这是因为+也表示字符串连接操作;
  • 二元操作符-会把两个操作数转换为数字类型;[35]
  • 一元操作符,包括+-,都会把操作数转换为数字。

下列为变量转换为字符串的例子:

  • 字符串类型不变;
  • 数字会转换为其字符串表示;
  • 数组的元素会转换为字符串,然后连接成通过逗号,分隔的长字符串;
  • 其它对象会转换为[object Object],其中Object中该对象的构造函数名。[36]

类型的隐藏转换是JavaScript受到批评的原因之一,因为隐藏转换增加了规则的复杂度和发生错误的可能性。[37]

更多信息 左操作数, 操作符 ...
JavaScript中的隐式转换[38][39]:
左操作数 操作符 右操作数 结果
[](空数组) + [](空数组) ""(空字符串)
[](空数组) + {}(空对象) "[object Object]"(字符串)
false(布尔值) + ""(空字符串) "false"(字符串)
"123"(字符串) + 1(数字) "1231"(字符串)
"123"(字符串) - 1(数字) 122 (数字)
关闭
faviconfaviconfaviconfavicon
7 sources

动态化

favicon
1 sources

类型

JavaScript是动态类型的语言,其类型与值而不是与变量相关联。例如变量可以为数字,随后又可被赋值为字符串。JavaScript提供了包括鸭子类型在内的方法来检测变量类型。

运行时估值

JavaScript提供eval()函数,可以在运行时直接执行JavaScript语句。[40]

favicon
1 sources

基于原型的面向对象

在JavaScript中,对象是关联数组,通过原型(prototype,见下)进行扩充。每一个字符串键值提供对象的一个属性的名称,可以通过使用点号(obj.x)或使用方括号(obj['x'])这两种效果相同的方式来访问。属性可以在运行时添加、重定义或删除。一个对象的大多数属性(包括来自原型继承链的属性)都可以通过 for...in循环访问。[41]

faviconfaviconfaviconfaviconfavicon
8 sources

原型

JavaScript使用原型,而许多其它面向对象语言使用类用于实现继承。原型使得在JavaScript中模拟基于类的面向对象特征变成可能。[42]

favicon
1 sources

函数作为对象构造器

函数在JavaScript中兼作为对象构造函数。在函数调用前加上new会创建一个原型的实例,并继承来自构造函数的属性和方法(包括来自Object原型)。[43]ECMAScript 5提供Object.create方法,可以显式地创建实例而不是自动从Object继承。[44]构造函数的prototype属性决定了用于新对象的内部原型。可以通过修改构造函数的原型的方法来为对象添加新的方法,也可以修改JavaScript的内部对象的原型(如ArrayObject)。尽管可以这么做,但对Object原型进行修改并不是好的做法。因为大多数JavaScript对象都会从Object继承,且不会希望其原型被修改。[45]

faviconfaviconfavicon
3 sources

函数作为方法

和大多数面向对象的语言不同,在JavaScript中函数定义和方法定义没有明显区别。唯一的区别在于调用时:当函数被作为方法调用时,函数的this会指向调用此函数的对象。[46]

favicon
1 sources

传统的类定义与使用格式

ECMAScript 2015中加入了对classextends关键字的支持,使得类的定义与继承更类似于其他的面向对象语言,同时也更易使用。[47][48]

faviconfavicon
2 sources

函数式

在JavaScript中,函数是一等的,函数也被认为是对象。因此,函数可以有属性和方法,例如call()bind等。[49]嵌套函数指定义于其它函数内部的函数,在外部函数被调用时,嵌套函数会被创建。另外,嵌套函数是一个闭包,在外部函数的作用域(包括常量,局部变量和参数)都成为内部函数状态的一部分,甚至在外部函数执行完毕后,内部函数的状态依然保留[50]。JavaScript同时也支持匿名函数[51]

faviconfavicon
3 sources

其它

favicon
4 sources

运行时环境

JavaScript通常依赖于运行时环境(例如浏览器)来提供对象和方法,脚本可以通过这些对象和方法与环境(例如网页DOM)进行交互。它还依赖于运行时环境来提供包或导入脚本(例如HTML<script>元素)的功能。这本身不是语言功能,但在大多数JavaScript实现中很常见。

承诺

JavaScript的“承诺”(Promise)是一种编程模型,它允许表示可能在未来某个时间点完成或失败的值。承诺被设计为一个解决异步编程中所遇到的问题的更简洁和一致的方法。相比于传统的回调函数,它为处理异步操作提供了一个更清晰的方式。

一个承诺有三种状态: 待定(Pending): 初始状态,既不是成功也不是失败。 已履行(Fulfilled): 意味着操作成功完成。 已拒绝(Rejected): 意味着操作失败。 承诺的主要特点是它们的状态一旦改变(从待定到已履行或已拒绝),就不能再次改变。这使得承诺成为一个可靠的资讯来源,无论它们是否成功。

在JavaScript中,你可以使用then()方法来附加回调函数,这些回调函数会在承诺达到已履行或已拒绝状态时执行。还有一个catch()方法,专门用于处理被拒绝的承诺。为了使链式调用更简洁,then()catch()都会返回一个新的承诺。ES6(ECMAScript 2015)引入了原生的Promise对象,从此承诺成为JavaScript标准的一部分。此后的版本还提供了额外的实用方法,例如Promise.all()Promise.race(),用于处理多个承诺。

异步

JavaScript一般来说是单线程的。[52]为了并发地处理事件,JavaScript程序输入或输出时使用事件回调函数执行。这意味着JavaScript可以在等待数据库查询返回资讯之前处理滑鼠单击。ECMAScript 2015引入了Promise用于处理异步事件,其可以使得传统的基于回调的异步代码更加清晰简单。[53][54]

favicon
3 sources

变长参数

JavaScript中函数参数的长度是可变的,在函数内部可以通过arguments对象访问这些参数。[55]

favicon
1 sources

编程

JavaScript是一门脚本语言,其原始码在客户端执行前不需经过编译,而是将文本格式的字符代码发送给浏览器,由浏览器解释执行。解释型语言的弱点是安全性较差,而且在JavaScript中,如果一条语句执行不了,通常它下面的语言也就无法执行。解决办法是使用异常处理try {} catch () {}

console.log("a");  //这是正确的
console.log("b");  //这是正确的
console.logg("c"); //这是错误的,并且到这里会停下来
console.log("d");  //这是正确的
console.log("e");  //这是正确的

/* 解決辦法 */
try { console.log("a"); } catch (e) {}  //这是正确的
try { console.log("b"); } catch (e) {}  //这是正确的
try { console.logg("c"); } catch (e) {} //这是错误的,但是到这里不会停下来,而是跳过
try { console.log("d"); } catch (e) {}  //这是正确的
try { console.log("e"); } catch (e) {}  //这是正确的

JavaScript被归类为解释型语言,因为主流的引擎都是每次执行时加载代码并解译。V8是将所有代码解译后再开始执行,其他引擎则是逐行解译(SpiderMonkey会将解译过的指令暂存,以提高性能,称为即时编译)。但由于V8的核心部分多数用JavaScript撰写(而SpiderMonkey是用C++),因此在不同的测试上,两者性能互有优劣。

与其相对应的是编译语言(如C语言),以编译语言编写的程序在执行之前必须经过编译,将代码编译为机器代码,才可以执行。

示例

以下是一个简单的JavaScript Hello World

<!DOCTYPE HTML>
<html>
    <head>
    <title>简单的JavaScript Hello World</title>
        <script>
            document.write("Hello, world!");   // 直接插入页面中
            alert("Hello, world!");            // 弹窗显示
            console.log("Hello, world!");      // 在控制台(console)里显示,需要先开启开发工具控制台
        </script>
    </head>
    <body>
    HTML内容……
    </body>
</html>

或是在浏览器地址栏中使用javascript:,以交互方式表示:

javascript:alert("Hello world!");

版本

JavaScript最初开发于1996年,被使用于Netscape Navigator网页浏览器。同年微软在Internet Explorer发布了一个实现。由于商标问题,这项实现被命名为JScript。1997年,JavaScript已经被网景公司提交给ECMA制定为标准,称之为ECMAScript,标准编号ECMA-262。使用显式版本号声明对语言的引用,作为一项Mozilla的特性,已在较新版本中被移除(至少为Firefox 59)。Firefox 4是最后一个需要显式地在引用时声明明确版本号的版本(1.8.5)。

下列表格的资讯基于多个参考来源[56][57][58][59]

更多信息 版本, 发布日期 ...
版本 发布日期 基于 Netscape
Navigator
Mozilla
Firefox
Internet
Explorer
Opera Safari Google
Chrome
旧版本,不再支持: 1.0 1996年3月 2.0 3.0
旧版本,不再支持: 1.1 1996年8月 3.0
旧版本,不再支持: 1.2 1997年6月 4.0 - 4.05 3
旧版本,不再支持: 1.3 1998年10月 ECMA-262 1st + 2nd edition 4.06 - 4.7x 4.0 5[60]
旧版本,不再支持: 1.4 Netscape
Server
6
旧版本,不再支持: 1.5 2000年11月 ECMA-262 3rd edition 6.0 1.0 5.5(JScript 5.5)
6(JScript 5.6)
7(JScript 5.7)
8(JScript 5.8)
7.0 3.0-5 1.0 - 10.0.666
旧版本,不再支持: 1.6 2005年11月 1.5 + Array extras + Array and string generics + E4X 1.5
旧版本,不再支持: 1.7 2006年10月 1.6 + Pythonic generators[61] + Iterators + Let 2.0 28.0.1500.95
旧版本,不再支持: 1.8 2008年6月 1.7 + Generator expressions + Expression closures 3.0 11.50
旧版本,不再支持: 1.8.1 1.8 + native JSON support + Minor updates 3.5
旧版本,不再支持: 1.8.2 2009年6月22日 1.8.1 + Minor updates 3.6
旧版本,不再支持: 1.8.5 2010年7月27日 1.8.2 + New features for ECMA-262 5th edition compliance 4.0
关闭
faviconfaviconfaviconfaviconfavicon
6 sources

参见

参考文献

外部链接

Wikiwand in your browser!

Seamless Wikipedia browsing. On steroids.

Every time you click a link to Wikipedia, Wiktionary or Wikiquote in your browser's search results, it will show the modern Wikiwand interface.

Wikiwand extension is a five stars, simple, with minimum permission required to keep your browsing private, safe and transparent.