读懂 ECMAScript 规格

概述

规格文件是计算机语言的官方标准,详细描述语法规则和实现方法。javascript

通常来讲,没有必要阅读规格,除非你要写编译器。由于规格写得很是抽象和精炼,又缺少实例,不容易理解,并且对于解决实际的应用问题,帮助不大。可是,若是你遇到疑难的语法问题,实在找不到答案,这时能够去查看规格文件,了解语言标准是怎么说的。规格是解决问题的“最后一招”。java

这对 JavaScript 语言颇有必要。由于它的使用场景复杂,语法规则不统一,例外不少,各类运行环境的行为不一致,致使奇怪的语法问题层出不穷,任何语法书都不可能囊括全部状况。查看规格,不失为一种解决语法问题的最可靠、最权威的终极方法。git

本章介绍如何读懂 ECMAScript 6 的规格文件。github

ECMAScript 6 的规格,能够在 ECMA 国际标准组织的官方网站(www.ecma-international.org/ecma-262/6.0/)免费下载和在线阅读。算法

这个规格文件至关庞大,一共有 26 章,A4 打印的话,足足有 545 页。它的特色就是规定得很是细致,每个语法行为、每个函数的实现都作了详尽的清晰的描述。基本上,编译器做者只要把每一步翻译成代码就能够了。这很大程度上,保证了全部 ES6 实现都有一致的行为。数组

ECMAScript 6 规格的 26 章之中,第 1 章到第 3 章是对文件自己的介绍,与语言关系不大。第 4 章是对这门语言整体设计的描述,有兴趣的读者能够读一下。第 5 章到第 8 章是语言宏观层面的描述。第 5 章是规格的名词解释和写法的介绍,第 6 章介绍数据类型,第 7 章介绍语言内部用到的抽象操做,第 8 章介绍代码如何运行。第 9 章到第 26 章介绍具体的语法。数据结构

对于通常用户来讲,除了第 4 章,其余章节都涉及某一方面的细节,不用通读,只要在用到的时候,查阅相关章节便可。app

术语

ES6 规格使用了一些专门的术语,了解这些术语,能够帮助你读懂规格。本节介绍其中的几个。ide

抽象操做

所谓”抽象操做“(abstract operations)就是引擎的一些内部方法,外部不能调用。规格定义了一系列的抽象操做,规定了它们的行为,留给各类引擎本身去实现。函数

举例来讲,Boolean(value)的算法,第一步是这样的。

  1. Let b be ToBoolean(value).

这里的ToBoolean就是一个抽象操做,是引擎内部求出布尔值的算法。

许多函数的算法都会屡次用到一样的步骤,因此 ES6 规格将它们抽出来,定义成”抽象操做“,方便描述。

Record 和 field

ES6 规格将键值对(key-value map)的数据结构称为 Record,其中的每一组键值对称为 field。这就是说,一个 Record 由多个 field 组成,而每一个 field 都包含一个键名(key)和一个键值(value)。

[[Notation]]

ES6 规格大量使用[[Notation]]这种书写法,好比[[Value]][[Writable]][[Get]][[Set]]等等。它用来指代 field 的键名。

举例来讲,obj是一个 Record,它有一个Prototype属性。ES6 规格不会写obj.Prototype,而是写obj.[[Prototype]]。通常来讲,使用[[Notation]]这种书写法的属性,都是对象的内部属性。

全部的 JavaScript 函数都有一个内部属性[[Call]],用来运行该函数。

F.[[Call]](V, argumentsList) 

上面代码中,F是一个函数对象,[[Call]]是它的内部方法,F.[[call]]()表示运行该函数,V表示[[Call]]运行时this的值,argumentsList则是调用时传入函数的参数。

Completion Record

每个语句都会返回一个 Completion Record,表示运行结果。每一个 Completion Record 有一个[[Type]]属性,表示运行结果的类型。

[[Type]]属性有五种可能的值。

  • normal
  • return
  • throw
  • break
  • continue

若是[[Type]]的值是normal,就称为 normal completion,表示运行正常。其余的值,都称为 abrupt completion。其中,开发者只须要关注[[Type]]throw的状况,即运行出错;breakcontinuereturn这三个值都只出如今特定场景,能够不用考虑。

抽象操做的标准流程

抽象操做的运行流程,通常是下面这样。

  1. Let resultCompletionRecord be AbstractOp().
  2. If resultCompletionRecord is an abrupt completion, return resultCompletionRecord.
  3. Let result be resultCompletionRecord.[[Value]].
  4. return result.

上面的第一步是调用抽象操做AbstractOp(),获得resultCompletionRecord,这是一个 Completion Record。第二步,若是这个 Record 属于 abrupt completion,就将resultCompletionRecord返回给用户。若是此处没有返回,就表示运行结果正常,所得的值存放在resultCompletionRecord.[[Value]]属性。第三步,将这个值记为result。第四步,将result返回给用户。

ES6 规格将这个标准流程,使用简写的方式表达。

  1. Let result be AbstractOp().
  2. ReturnIfAbrupt(result).
  3. return result.

这个简写方式里面的ReturnIfAbrupt(result),就表明了上面的第二步和第三步,即若是有报错,就返回错误,不然取出值。

甚至还有进一步的简写格式。

  1. Let result be ? AbstractOp().
  2. return result.

上面流程的?,就表明AbstractOp()可能会报错。一旦报错,就返回错误,不然取出值。

除了?,ES 6 规格还使用另外一个简写符号!

  1. Let result be ! AbstractOp().
  2. return result.

上面流程的!,表明AbstractOp()不会报错,返回的必定是 normal completion,老是能够取出值。

相等运算符

下面经过一些例子,介绍如何使用这份规格。

相等运算符(==)是一个很让人头痛的运算符,它的语法行为多变,不符合直觉。这个小节就看看规格怎么规定它的行为。

请看下面这个表达式,请问它的值是多少。

0 == null 

若是你不肯定答案,或者想知道语言内部怎么处理,就能够去查看规格,7.2.12 小节是对相等运算符(==)的描述。

规格对每一种语法行为的描述,都分红两部分:先是整体的行为描述,而后是实现的算法细节。相等运算符的整体描述,只有一句话。

“The comparison x == y, where x and y are values, produces true or false.”

上面这句话的意思是,相等运算符用于比较两个值,返回truefalse

下面是算法细节。

  1. ReturnIfAbrupt(x).
  2. ReturnIfAbrupt(y).
  3. If Type(x) is the same as Type(y), then
    1. Return the result of performing Strict Equality Comparison x === y.
  4. If x is null and y is undefined, return true.
  5. If x is undefined and y is null, return true.
  6. If Type(x) is Number and Type(y) is String, return the result of the comparison x == ToNumber(y).
  7. If Type(x) is String and Type(y) is Number, return the result of the comparison ToNumber(x) == y.
  8. If Type(x) is Boolean, return the result of the comparison ToNumber(x) == y.
  9. If Type(y) is Boolean, return the result of the comparison x == ToNumber(y).
  10. If Type(x) is either String, Number, or Symbol and Type(y) is Object, then return the result of the comparison x == ToPrimitive(y).
  11. If Type(x) is Object and Type(y) is either String, Number, or Symbol, then return the result of the comparison ToPrimitive(x) == y.
  12. Return false.

上面这段算法,一共有 12 步,翻译以下。

  1. 若是x不是正常值(好比抛出一个错误),中断执行。
  2. 若是y不是正常值,中断执行。
  3. 若是Type(x)Type(y)相同,执行严格相等运算x === y
  4. 若是xnullyundefined,返回true
  5. 若是xundefinedynull,返回true
  6. 若是Type(x)是数值,Type(y)是字符串,返回x == ToNumber(y)的结果。
  7. 若是Type(x)是字符串,Type(y)是数值,返回ToNumber(x) == y的结果。
  8. 若是Type(x)是布尔值,返回ToNumber(x) == y的结果。
  9. 若是Type(y)是布尔值,返回x == ToNumber(y)的结果。
  10. 若是Type(x)是字符串或数值或Symbol值,Type(y)是对象,返回x == ToPrimitive(y)的结果。
  11. 若是Type(x)是对象,Type(y)是字符串或数值或Symbol值,返回ToPrimitive(x) == y的结果。
  12. 返回false

因为0的类型是数值,null的类型是 Null(这是规格4.3.13 小节的规定,是内部 Type 运算的结果,跟typeof运算符无关)。所以上面的前 11 步都得不到结果,要到第 12 步才能获得false

0 == null // false 

数组的空位

下面再看另外一个例子。

const a1 = [undefined, undefined, undefined]; const a2 = [, , ,]; a1.length // 3 a2.length // 3 a1[0] // undefined a2[0] // undefined a1[0] === a2[0] // true 

上面代码中,数组a1的成员是三个undefined,数组a2的成员是三个空位。这两个数组很类似,长度都是 3,每一个位置的成员读取出来都是undefined

可是,它们实际上存在重大差别。

0 in a1 // true 0 in a2 // false a1.hasOwnProperty(0) // true a2.hasOwnProperty(0) // false Object.keys(a1) // ["0", "1", "2"] Object.keys(a2) // [] a1.map(n => 1) // [1, 1, 1] a2.map(n => 1) // [, , ,] 

上面代码一共列出了四种运算,数组a1a2的结果都不同。前三种运算(in运算符、数组的hasOwnProperty方法、Object.keys方法)都说明,数组a2取不到属性名。最后一种运算(数组的map方法)说明,数组a2没有发生遍历。

为何a1a2成员的行为不一致?数组的成员是undefined或空位,到底有什么不一样?

规格的12.2.5 小节《数组的初始化》给出了答案。

“Array elements may be elided at the beginning, middle or end of the element list. Whenever a comma in the element list is not preceded by an AssignmentExpression (i.e., a comma at the beginning or after another comma), the missing array element contributes to the length of the Array and increases the index of subsequent elements. Elided array elements are not defined. If an element is elided at the end of an array, that element does not contribute to the length of the Array.”

翻译以下。

"数组成员能够省略。只要逗号前面没有任何表达式,数组的length属性就会加 1,而且相应增长其后成员的位置索引。被省略的成员不会被定义。若是被省略的成员是数组最后一个成员,则不会致使数组length属性增长。”

上面的规格说得很清楚,数组的空位会反映在length属性,也就是说空位有本身的位置,可是这个位置的值是未定义,即这个值是不存在的。若是必定要读取,结果就是undefined(由于undefined在 JavaScript 语言中表示不存在)。

这就解释了为何in运算符、数组的hasOwnProperty方法、Object.keys方法,都取不到空位的属性名。由于这个属性名根本就不存在,规格里面没说要为空位分配属性名(位置索引),只说要为下一个元素的位置索引加 1。

至于为何数组的map方法会跳过空位,请看下一节。

数组的 map 方法

规格的22.1.3.15 小节定义了数组的map方法。该小节先是整体描述map方法的行为,里面没有提到数组空位。

后面的算法描述是这样的。

  1. Let O be ToObject(this value).
  2. ReturnIfAbrupt(O).
  3. Let len be ToLength(Get(O, "length")).
  4. ReturnIfAbrupt(len).
  5. If IsCallable(callbackfn) is false, throw a TypeError exception.
  6. If thisArg was supplied, let T be thisArg; else let T be undefined.
  7. Let A be ArraySpeciesCreate(O, len).
  8. ReturnIfAbrupt(A).
  9. Let k be 0.
  10. Repeat, while k < len
    1. Let Pk be ToString(k).
    2. Let kPresent be HasProperty(O, Pk).
    3. ReturnIfAbrupt(kPresent).
    4. If kPresent is true, then
      1. Let kValue be Get(O, Pk).
      2. ReturnIfAbrupt(kValue).
      3. Let mappedValue be Call(callbackfn, T, «kValue, k, O»).
      4. ReturnIfAbrupt(mappedValue).
      5. Let status be CreateDataPropertyOrThrow (A, Pk, mappedValue).
      6. ReturnIfAbrupt(status).
    5. Increase k by 1.
  11. Return A.

翻译以下。

  1. 获得当前数组的this对象
  2. 若是报错就返回
  3. 求出当前数组的length属性
  4. 若是报错就返回
  5. 若是 map 方法的参数callbackfn不可执行,就报错
  6. 若是 map 方法的参数之中,指定了this,就让T等于该参数,不然Tundefined
  7. 生成一个新的数组A,跟当前数组的length属性保持一致
  8. 若是报错就返回
  9. 设定k等于 0
  10. 只要k小于当前数组的length属性,就重复下面步骤
    1. 设定Pk等于ToString(k),即将K转为字符串
    2. 设定kPresent等于HasProperty(O, Pk),即求当前数组有没有指定属性
    3. 若是报错就返回
    4. 若是kPresent等于true,则进行下面步骤
      1. 设定kValue等于Get(O, Pk),取出当前数组的指定属性
      2. 若是报错就返回
      3. 设定mappedValue等于Call(callbackfn, T, «kValue, k, O»),即执行回调函数
      4. 若是报错就返回
      5. 设定status等于CreateDataPropertyOrThrow (A, Pk, mappedValue),即将回调函数的值放入A数组的指定位置
      6. 若是报错就返回
    5. k增长 1
  11. 返回A

仔细查看上面的算法,能够发现,当处理一个全是空位的数组时,前面步骤都没有问题。进入第 10 步中第 2 步时,kPresent会报错,由于空位对应的属性名,对于数组来讲是不存在的,所以就会返回,不会进行后面的步骤。

const arr = [, , ,]; arr.map(n => { console.log(n); return 1; }) // [, , ,] 

上面代码中,arr是一个全是空位的数组,map方法遍历成员时,发现是空位,就直接跳过,不会进入回调函数。所以,回调函数里面的console.log语句根本不会执行,整个map方法返回一个全是空位的新数组。

V8 引擎对map方法的实现以下,能够看到跟规格的算法描述彻底一致。

function ArrayMap(f, receiver) { CHECK_OBJECT_COERCIBLE(this, "Array.prototype.map");  // Pull out the length so that modifications to the length in the  // loop will not affect the looping and side effects are visible. var array = TO_OBJECT(this); var length = TO_LENGTH_OR_UINT32(array.length); return InnerArrayMap(f, receiver, array, length); } function InnerArrayMap(f, receiver, array, length) { if (!IS_CALLABLE(f)) throw MakeTypeError(kCalledNonCallable, f); var accumulator = new InternalArray(length); var is_array = IS_ARRAY(array); var stepping = DEBUG_IS_STEPPING(f); for (var i = 0; i < length; i++) { if (HAS_INDEX(array, i, is_array)) { var element = array[i];  // Prepare break slots for debugger step in. if (stepping) %DebugPrepareStepInIfStepping(f); accumulator[i] = %_Call(f, receiver, element, i, array); } } var result = new GlobalArray(); %MoveArrayContents(accumulator, result); return result; }
相关文章
相关标签/搜索