ImageVerifierCode 换一换
格式:DOC , 页数:8 ,大小:116.50KB ,
资源ID:7035313      下载积分:20 文币
快捷下载
登录下载
邮箱/手机:
温馨提示:
快捷下载时,用户名和密码都是您填写的邮箱或者手机号,方便查询和重复下载(系统自动生成)。 如填写123,账号就是123,密码也是123。
特别说明:
请自助下载,系统不会自动发送文件的哦; 如果您已付费,想二次下载,请登录后访问:我的下载记录
支付方式: 支付宝    微信支付   
验证码:   换一换

加入VIP,免费下载
 

温馨提示:由于个人手机设置不同,如果发现不能下载,请复制以下地址【https://www.wenkunet.com/d-7035313.html】到电脑端继续下载(重复下载不扣费)。

已注册用户请登录:
账号:
密码:
验证码:   换一换
  忘记密码?
三方登录: 微信登录   QQ登录   微博登录 

下载须知

1: 本站所有资源如无特殊说明,都需要本地电脑安装OFFICE2007和PDF阅读器。
2: 试题试卷类文档,如果标题没有明确说明有答案则都视为没有答案,请知晓。
3: 文件的所有权益归上传用户所有。
4. 未经权益所有人同意不得将文件中的内容挪作商业或盈利用途。
5. 本站仅提供交流平台,并不能对任何下载内容负责。
6. 下载文件中如有侵权或不适当内容,请与我们联系,我们立即纠正。
7. 本站不保证下载资源的准确性、安全性和完整性, 同时也不承担用户因使用这些下载资源对自己和他人造成任何形式的伤害或损失。

版权提示 | 免责声明

本文(13读懂ES6规格.doc)为本站会员(nanchangxurui)主动上传,文库网仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对上载内容本身不做任何修改或编辑。 若此文所含内容侵犯了您的版权或隐私,请立即通知文库网(发送邮件至13560552955@163.com或直接QQ联系客服),我们立即给予删除!

13读懂ES6规格.doc

1、概述规格文件是计算机语言的官方标准,详细描述语法规则和实现方法。一般来说,没有必要阅读规格,除非你要写编译器。因为规格写得非常抽象和精炼,又缺乏实例,不容易理解,而且对于解决实际的应用问题,帮助不大。但是,如果你遇到疑难的语法问题,实在找不到答案,这时可以去查看规格文件,了解语言标准是怎么说的。规格是解决问题的“最后一招”。这对JavaScript语言很有必要。因为它的使用场景复杂,语法规则不统一,例外很多,各种运行环境的行为不一致,导致奇怪的语法问题层出不穷,任何语法书都不可能囊括所有情况。查看规格,不失为一种解决语法问题的最可靠、最权威的终极方法。本章介绍如何读懂ECMAScript 6的

2、规格文件。ECMAScript 6的规格,可以在ECMA国际标准组织的官方网站(www.ecma-international.org/ecma-262/6.0/)免费下载和在线阅读。这个规格文件相当庞大,一共有26章,A4打印的话,足足有545页。它的特点就是规定得非常细致,每一个语法行为、每一个函数的实现都做了详尽的清晰的描述。基本上,编译器作者只要把每一步翻译成代码就可以了。这很大程度上,保证了所有ES6实现都有一致的行为。ECMAScript 6规格的26章之中,第1章到第3章是对文件本身的介绍,与语言关系不大。第4章是对这门语言总体设计的描述,有兴趣的读者可以读一下。第5章到第8章是语

3、言宏观层面的描述。第5章是规格的名词解释和写法的介绍,第6章介绍数据类型,第7章介绍语言内部用到的抽象操作,第8章介绍代码如何运行。第9章到第26章介绍具体的语法。对于一般用户来说,除了第4章,其他章节都涉及某一方面的细节,不用通读,只要在用到的时候,查阅相关章节即可。下面通过一些例子,介绍如何使用这份规格。相等运算符相等运算符(=)是一个很让人头痛的运算符,它的语法行为多变,不符合直觉。这个小节就看看规格怎么规定它的行为。请看下面这个表达式,请问它的值是多少。0 = null如果你不确定答案,或者想知道语言内部怎么处理,就可以去查看规格,7.2.12小节是对相等运算符(=)的描述。规格对每一

4、种语法行为的描述,都分成两部分:先是总体的行为描述,然后是实现的算法细节。相等运算符的总体描述,只有一句话。“The comparisonx = y, wherexandyare values, producestrueorfalse.”上面这句话的意思是,相等运算符用于比较两个值,返回true或false。下面是算法细节。1. ReturnIfAbrupt(x).2. ReturnIfAbrupt(y).3. IfType(x)is the same asType(y), thenReturn the result of performing Strict Equality Comparis

5、onx = y.4. Ifxisnullandyisundefined, returntrue.5. Ifxisundefinedandyisnull, returntrue.6. IfType(x)is Number andType(y)is String,return the result of the comparisonx = ToNumber(y).7. IfType(x)is String andType(y)is Number,return the result of the comparisonToNumber(x) = y.8. IfType(x)is Boolean, re

6、turn the result of the comparisonToNumber(x) = y.9. IfType(y)is Boolean, return the result of the comparisonx = ToNumber(y).10. IfType(x)is either String, Number, or Symbol andType(y)is Object, thenreturn the result of the comparisonx = ToPrimitive(y).11. IfType(x)is Object andType(y)is either Strin

7、g, Number, or Symbol, thenreturn the result of the comparisonToPrimitive(x) = y.12. Returnfalse.上面这段算法,一共有12步,翻译如下。1. 如果x不是正常值(比如抛出一个错误),中断执行。2. 如果y不是正常值,中断执行。3. 如果Type(x)与Type(y)相同,执行严格相等运算x = y。4. 如果x是null,y是undefined,返回true。5. 如果x是undefined,y是null,返回true。6. 如果Type(x)是数值,Type(y)是字符串,返回x = ToNumber

8、(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

9、.3.13小节的规定,是内部Type运算的结果,跟typeof运算符无关)。因此上面的前11步都得不到结果,要到第12步才能得到false。0 = null / false数组的空位下面再看另一个例子。const a1 = undefined, undefined, undefined;const a2 = , , ,;a1.length / 3a2.length / 3a10 / undefineda20 / undefineda10 = a20 / true上面代码中,数组a1的成员是三个undefined,数组a2的成员是三个空位。这两个数组很相似,长度都是3,每个位置的成员读取出来都是

10、undefined。但是,它们实际上存在重大差异。0 in a1 / true0 in a2 / falsea1.hasOwnProperty(0) / truea2.hasOwnProperty(0) / falseObject.keys(a1) / 0, 1, 2Object.keys(a2) / a1.map(n = 1) / 1, 1, 1a2.map(n = 1) / , , ,上面代码一共列出了四种运算,数组a1和a2的结果都不一样。前三种运算(in运算符、数组的hasOwnProperty方法、Object.keys方法)都说明,数组a2取不到属性名。最后一种运算(数组的map方

11、法)说明,数组a2没有发生遍历。为什么a1与a2成员的行为不一致?数组的成员是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

12、), 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.”翻译如下。数组成员可以省略。只要逗号前面没有任何表达式,数组的lengt

13、h属性就会加1,并且相应增加其后成员的位置索引。被省略的成员不会被定义。如果被省略的成员是数组最后一个成员,则不会导致数组length属性增加。”上面的规格说得很清楚,数组的空位会反映在length属性,也就是说空位有自己的位置,但是这个位置的值是未定义,即这个值是不存在的。如果一定要读取,结果就是undefined(因为undefined在JavaScript语言中表示不存在)。这就解释了为什么in运算符、数组的hasOwnProperty方法、Object.keys方法,都取不到空位的属性名。因为这个属性名根本就不存在,规格里面没说要为空位分配属性名(位置索引),只说要为下一个元素的位置索

14、引加1。至于为什么数组的map方法会跳过空位,请看下一节。数组的map方法规格的22.1.3.15小节定义了数组的map方法。该小节先是总体描述map方法的行为,里面没有提到数组空位。后面的算法描述是这样的。1. LetObeToObject(this value).2. ReturnIfAbrupt(O).3. LetlenbeToLength(Get(O, length).4. ReturnIfAbrupt(len).5. IfIsCallable(callbackfn)isfalse, throw a TypeError exception.6. IfthisArgwas supplie

15、d, letTbethisArg; else letTbeundefined.7. LetAbeArraySpeciesCreate(O, len).8. ReturnIfAbrupt(A).9. Letkbe 0.10. Repeat, whilek console.log(n); return 1;) / , , ,上面代码中,arr是一个全是空位的数组,map方法遍历成员时,发现是空位,就直接跳过,不会进入回调函数。因此,回调函数里面的console.log语句根本不会执行,整个map方法返回一个全是空位的新数组。V8引擎对map方法的实现如下,可以看到跟规格的算法描述完全一致。func

16、tion 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 InnerArr

17、ayMap(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 = arrayi; / Prepare break slots for debugger step in. if (stepping) %DebugPrepareStepInIfStepping(f); accumulatori = %_Call(f, receiver, element, i, array); var result = new GlobalArray(); %MoveArrayContents(accumulator, result); return result;

本站链接:文库   一言   我酷   合作


客服QQ:2549714901微博号:文库网官方知乎号:文库网

经营许可证编号: 粤ICP备2021046453号世界地图

文库网官网©版权所有2025营业执照举报