《ES标准入门》&《UNDERSTANDING ECMACH…

2018-06-24 01:52:31来源:未知 阅读 ()

新老客户大回馈,云服务器低至5折



### 前言
*这两本书应该是目前ES6相关的比较好的了,网上有电子版本(文末有链接)。不过我买了书看,哈哈。这两篇摘录笔记分为上下两部分,本文是上半部分(1-6章),摘录了两本书里一些比较有用的知识点。*

### 目录

> #####1. 块级作用域绑定
> #####2. 字符串与正则表达式
> #####3. 函数的扩展
> #####4. 数组的扩展
> #####5. 对象的扩展
  #####6. 集合(Set、Map)
> 7 . Symbol和Symbol属性

> 8 . Javascript中的类

> 9 . Promise、Generator函数、Async函数

> 10 . 代理(Proxy)和反射(Reflection)API

> 11 . 修饰器

> 12 . Module


### 一、块级作用域绑定

块级声明用于声明在指定块级作用域之外无法访问的变量。块级作用域存在于:1、函数内部;2、块中(字符'{'和'}'之间的区域)。Es6 中存在的两种变量声明就是Let和Const声明。

值得摘录的有以下六点:

##### 1. 不能重复声明

    var count = 9; 
    //抛出语法错误
    let count = 7;

##### 2. Const常量声明必须进行初始化

    const age = 8;
    //语法错误:常量未初始化
    const name;

##### 3. 特殊的For循环
For循环设置循环变量的那部分是一个父作用域,而循环体内部是一个单独的子作用域。

    for(let i = 0;i< 3;i++){
        let i = 'can not change';
        console.log(i);//输出'can not change'
    }

##### 4. 暂时性死区
在区块中存在let和const命令,则这个区块对这些命令声明的变量从一开始就形成封闭的作用域。只要在声明之前就使用这些变量,就会报错。

    情况一:
    var tmp = 123;
    if(true){
        tmp = 'abc';//ReferenceError
        let tmp;
    }
    
    情况二:
    function test(x = y,y = 2){
        return [x,y];
    }
    
    test();//报错

    情况三:
    let x = x;//ReferenceError: x is not defined

##### 5. Const声明对象,对象属性可变

##### 6. 彻底冻结函数

    constantize(obj) {//彻底冻结函数
        Object.freeze(obj);
        Object.keys(obj).forEach((key, i) => {
            if (typeof obj[key] === 'object') {
                constantize(obj[key]);
            }
        })
        obj.name = 777;//TypeError: Cannot add property name, object is not extensible
        return obj;
    }


### 二、字符串与正则表达式
Es6加强了对Unicode的支持,并且扩展了字符串对象。正则表达式则增加了修饰符和属性。

值得摘录的有以下几点:

##### 1. codePointAt()与String.fromCodePoint()方法
完全支持UTF-16,接受编码单元的位置而非字符位置作为参数,返回与字符串中给定位置对应的码位,即一个整数值:

    var text = "??a";

    console.log(text.charCodeAt(0));  //55362
    console.log(text.codePointAt(0)); //134017

    判断一个字符是2个字节还是4个字节组成:
    function is32Bit(c){
        return c.codePointAt(0) > 0xFFFF;
    }

    is32Bit("??"); // true
    is32Bit("a"); // false

    console.log(String.fromCodePoint(134071)); // "??"

##### 2. normalize()方法
Es6为字符串添加了一个Normalize()方法,它可以提供Unicode的标准化形式,接受一个参数,指明应用哪种Unicode标准化形式(NFC,NFD,NFKC,NFKD)。只需要记住,在对比字符串之前,一定要先把它们标准化为同一种形式:

    let normalized = values.map((text) => {
        return text.normalize();
    })

    console.log(this[normalize]('\u01D1') === this[normalize]('\u004F\u030C')) // true

##### 3. includes()、startsWith()、endsWith()、repeat()方法

    let tempArray = [1, "4", "uu", 5];
    let tempStr = "test123uuunit582";

    console.log(tempArray.includes(5)); //true
    console.log(tempStr.includes("test123", 1)); //从第二个位置进行匹配

    console.log(tempStr.startsWith("test")); //true
    console.log(tempStr.startsWith("test123", 0)); //从第一个位置进行匹配

    console.log(tempStr.endsWith("12",6)); //从第六个位置进行匹配
    console.log(tempStr.repeat(3));    //test123uuunit582test123uuunit582test123uuunit582

##### 4. 模板字符串
领域专用语言(DSL),可以生成、查询并操作其他语言里的内容,且可以免受注入攻击(XSS、SQL等)。

> A、去掉模板字符串中的换行


        $('#list').html(`
          <ul>
              <li>first</li>
              <li>second</li>
          </ul>
        `.trim());

> B、引用模板字符串本身


        //写法一
        let str = 'return ' + '`ni hao ${name}`';
        let func = new Function('name',str);
        let data = func('uct');
        //写法二
        let str2 = '(name) => `ni hao ${name}`';
        let func2 = eval.call(null,str2);
        let data2 = func2('uct');
    
        console.log(data + '  ' + data2);

> C、标签模板


        alert`123`;
        //等同于
        alert(123);
        var a = 1,b = 2;
        tag`Hello ${a + b} world ${a * b}`;
        //等同于
        tag(['Hello ',' world , ''],1,2);//第一个参数是数组,存放模板中没有变量替换的部分,后续参数是变量。

> D、String.raw()转义模板字符串



        String.raw`Hi\n${3+4}`;//"Hi\\n5"

[关于new Function][newFunction]

[关于eval][eval]

[newFunction]: https://developer.mozilla.org/zh-CN/docs/Web/JavaScript/Reference/Global_Objects/Function
[eval]:https://developer.mozilla.org/zh-CN/docs/Web/JavaScript/Reference/Global_Objects/eval

##### 5.U修饰符与Y修饰符
U修饰符,即为“Unicode模式”,用来正确处理大于\uFFFF的Unicode字符。Y修饰符,叫作“粘连”(Sticky)修饰符,它的后一次匹配都从上一次匹配成功的下一个位置开始。

> A、点(.)字符匹配(匹配除换行符外的所有字符)

        var s = "??";
        /^.$/.test(s); // false
        /^.$/u.test(s); // true
    
> B、返回字符串长度的函数


        function codePointLength(){
            var result = text.match(/[\s\S]/gu);
            return result ? result.length : 0;
        }

> C、y修饰符隐含了头部匹配的标志(^)


        const REGEX = /a/y;
        REGEX.lastIndex = 2;
        REGEX.exec("xaya"); // null
        REGEX.lastIndex = 3;
        const match = REGEX.exec("xaya");
        match.index; // 3
    

        
### 三、函数的扩展

值得摘录的有以下几点:

    
##### 1.函数参数的默认值

> A、参数变量默认声明

        function test(x = 1){
            let x = 2;//error
        }

> B、惰性求值
        let x = 10;
        function testLazy(p = x + 1){
            console.log(p);
        }
    
        testLazy(); // 11
        let x = 12;
        testLazy(); // 13

> C、非尾部设置默认值,参数不可省略

        function f(x = 1,y){
            return [x,y];
        }
        f(,1); //报错

> D、默认参数的临时死区TDZ

        function add(first = second,second){
            return first + second;
        }

        console.log(add(1,1));    // 2
        console.log(add(undefined,1));  // 抛出错误
        //当初次调用add()时,绑定first和second被添加到一个专属于函数参数的临时死区

        //表示调用add(1,1) 时的Javascript 代码
        let first = 1;
        let second = 2;
        //表示调用add(undefined,1) 时的Javascript 代码
        let first = second;
        let second = 1; //所以抛出错误;

        
##### 2.默认参数表达式
初次解析函数声明时不会调用getValue()方法,只有当调用add()函数且不传入第二个参数时才会调用:


        constructor() {
            this.value = 5;
        }

        getValue(){
            return this.value ++;
        }
    
        add(first,second = this.getValue()){
            console.log('plus: ', first + second );
        }
        ...
        Es6Function.add(1,1); // plus:  2
        Es6Function.add(1);    // plus:  6
        Es6Function.add(1);    // plus:  7
            


##### 3. 只要函数参数使用了默认值、解构赋值或者扩展运算符,那么函数内部就不能严格模式

        const doSomething = (a,b = a,...c) => { //error
            'use strict';
            //code 
        }

##### 4. 不定参数限制使用

> 每个函数最多只能声明一个不定参数,而且一定要放在所有参数的末尾。

        function any(a,...b,last){} //报错

> 不定参数不能用于对象字面量Setter中

        let object = {

            //语法错误,不可以在Setter中使用不定参数
            set name(...value){}
        }

> 取数组最大值

        let values = [25,34,11,99];
    
        console.log(Math.max(...values)); // 99


##### 5. 箭头函数

> A、箭头函数中的This值取决于该函数外部非箭头函数的this值,且不能通过call()、apply()或bind()方法来改变this值

        thisScope(){
            setTimeout(()=>{
                console.log('id: ',this.id);
            },100);
        }
        ...
        var id = 55;
          Es6Function.thisScope.call({id:100}); // id : 100

> B、函数体内不可民使用Arguments对象,该对象在函数体内不存在,可以用Rest参数(...)代替

        testArguments(){
            setTimeout(() => {
                console.log('args: ',arguments)
            }, 100);
        }
        ...
        Es6Function.testArguments(1,2,5,6); // args:  Arguments(4) [1, 2, 5, 6, callee: (...), Symbol(Symbol.iterator): ?]

##### 6. 尾调用优化

> 尾调用就是指某个函数的最后一步是调用另一个函数,只在严格模式下开启。

        "use strict";

        function doSomething(){
            //优化后,立即返回结果
            return doSomethingElse();
        }
        
        function doSomething2(){
            //无法优化,无返回
            doSomethingElse();
        }

> 非严格模式下实现尾递归优化,用“循环” 替换 “递归”

        function sum(x,y){
            if(y > 0){
                return sum( x + 1,y - 1);
            } else {
                return x;
            }
    
        }
        sum(1,1000000);// Uncaught RangeError: Maximum call stack size exceeded(...)

        //用蹦床函数可以将递归执行转为循环执行
        function trampoline(f){
            while(f && f instanceof Function){
                f = f();
            }
            return f;
        }

        //返回一个函数,然后执行该函数,而不是在函数里调用函数,避免递归,消除调用栈过大问题

### 四、数组的扩展

值得摘录的有以下几点:

##### 1. 扩展运算符(...)

扩展运算符背后调用的是遍历器接口(Symbol.iterator),如果一个对象没有部署该接口就无法转换

> 替代数组的 Apply方法 

        //Es5 的写法

        function f(x,y,z){}

        var args = [1,2,3];

        f.apply(null,args);

        //Es6 的写法

        function f(x,y,z){}

        var args = [1,2,3];

        f(...args);

> 合并数组

        Es5 合并数组

        var arr1 = [1,2];
        var arr2 = [3,4];
        var arr3 = [5];

        arr1.concat(arr2,arr3);

        Es6 合并数组

        [...arr1,...arr2,...arr3];

> 能够识别 32 位的 Unicode 字符的取字符串长度的方法

        function length(str){
            return [...str].length;
        }
    
        length('x\uD83D\uDE80y'); // 3

##### 3. Array.from()方法,用于将类似数组对象和可遍历对象(包括Set和Map结构)转为真正的数组
Array.from()方法除了支持遍历器接口转换外,还支持类数组对象(类数组对象的本质就是有Length属性)

        Array.from({length:3});
        // [undefined,undefined,undefined] 

> 映射转换,Array.from()的第二个参数用来遍历类数组对象并返回

        translate() {
            return Array.from(arguments, (value) => value + 1);
        }

        console.log(NewArray.translate(1,2,3)); // [2,3,4]

##### 4. 为所有数组添加新的方法


> Array.of(),为了弥补数组构造函数Array()的不足

        Array.of(3,11,8); // [3,11,8]
        Array.of(undefined); // [undefined] , 通过Vue 显示到页面则为 [null]
        Array(3); // [,,]

> copyWithin(),把数组内部指定位置成员复制到其他位置(会覆盖其他成员)

        //将3号位复制到0号位
        [1,2,3,4,5].copyWithin(0,3,4); //[ 4, 2, 3, 4, 5 ]
        // -2相当于3号位,-1相当于4号位
        [1,2,3,4,5].copyWithin(0,-2,-1); //[ 4, 2, 3, 4, 5 ]
        
> find(),返回符合条件的第一个(可能有多个)成员,如果没有就返回undefined

        testFind(){ // 返回 3
            return [1,2,3,4,5].find((value,index,arr)=>{
                return value > 2;
            })
        }

> findIndex(),返回第一个符合条件的成员的位置,如果没有就返回 -1

        testFindIndex(){ // 返回 -1
            return [1,2,3,4,5].findIndex((value,index,arr)=>{
                return value > 6;
            })
        }

> fill(),填充一个数组,接受第二第三个参数(起始位置和结束位置)

        testFill(){
            console.log(['a','b','c'].fill(8,1,2)); // ["a", 8, "c"]
            return new Array(3).fill(8); //[ 8, 8, 8 ]
        }

> includes(),返回一个布尔值,表示某个数组是否包含给定的值,它与indexOf 比起来更加直观,并且可以判断NaN的情况

        testIncludes(){
            return [1,2,NaN].includes(NaN); // true
        }

> entries(),keys(),values()方法,它们都返回一个遍历器对象

        testEntries(){
            for(let [key,value] of ['a','b','c'].entries()){
                console.log(key,value); // 0 "a"  1 "b"  2 "c"
            }
            let letter = ['a','b','c'];
            //手动调用遍历器对象的Next方法进行遍历
            let entries = letter.entries();
            console.log(entries.next().value); //[0, "a"]
            console.log(entries.next().value); //[1, "b"]
            console.log(entries.next().value); //[2, "c"]
        }

### 五、对象的扩展
ECMAScript6 中,为了使某些任务更易完成,在全局Object 对象上引入了一些新方法

值得摘录的有以下几点:

##### 1. 可计算属性名

        let suffix = " name";

        var person = {
            ["first" + suffix]: "Nicholas", // Es5 版本,这里的名称是不可以计算的
            ["last" + suffix]: "Zakas",
        }

        console.log(person["first name"]); // "Nicholas"
        console.log(person["last name"]);    // "Zakas"

##### 2. Object.is(),弥补全等(===)运算符的特殊情况判断不一致

        testObjectIs(){
            console.log(+0 == -0);             // true
            console.log(+0 === -0);         // true
            console.log(Object.is(+0 , -0)); // false
            console.log(NaN == NaN);         // false
            console.log(NaN === NaN);         // false
            console.log(Object.is(NaN , NaN)); // true
        }

##### 3. Object.assign(),只会复制对象自身可枚举属性,忽略enumerable为false的属性

        
> 多个对象复制具有同名属性,排位靠后的源对象会覆盖排位靠前的

        testObjectAssign(){
            let receiver = {};
            Object.assign(receiver,{
                type:'girl',
                name:'superman',
            },{
                type:'boy',
            })
            return receiver.type + " : " + receiver.name; // boy : superman
        }

> 对象浅复制(源对象某个属性是对象,只复制这个对象的引用)

        let target = { a: { b: 'c', d: 'e' } };
        let source = { a: { b: 'hello' } }
        console.log(Object.assign(target, source)); // {a: {b:'hello'}}
        // a属性被整体覆盖掉,不会得到 {a : {b:'hello',d:'e'}} 的结果

##### 3. Object.values()

> 只返回对象自身可遍历属性

        testObjectValues() { // p 的属性描述对象的 enumerable 默认是 false
            var obj = Object.create({}, { p: { value: 42 } });
            return Object.values(obj); // []
        }

> Object.values 会过滤属性名为Symbol值的属性

        console.log(Object.values({ [Symbol()]: 123, any: 'abc' })); // ['abc']

##### 3. Object.entries()

> 转换对象为真正的Map结构

        testObjectEntries(){
            let obj = { a: 'bar', b: 23 };
            let map = new Map(Object.entries(obj));
            console.log(map.get('a')); // bar
        }


### 六、集合(Set、Map)
Es6 中的Set和Map核心思想跟Java中的Set和Map类似,Set集合成员是唯一的,没有重复,Set本身是构造函数用来生成Set数据结构;Map类型是一种存储着许多键值对的有序列表,它的键名的判断是通过Object.is()方法来实现的(Set也是通过这个方法来判断两个值是否一致),所以 5 和字符串"5"会被判定为两种类型。

值得摘录的有以下几点:

##### 1. Set

> 添加元素用add()方法,删除用delete()方法,判断存在用has(),清空用clear()

        testSet() {
            let set = new Set();
            set.add(5);
            set.add('5');
            console.log(set.has(5)); // true
            set.delete(5);
            console.log(set.has(5)); // false
            console.log(set.size); // 1
            set.clear();
            console.log(set.has('5')); // false
            console.log(set.size); // 0
         }

> 数组去重

        testArrayUniq(){
            let set = new Set([1,2,3,3,3,3,4,4,4,4,5]);
            let arr = [...set];
            console.log(arr); //[ 1, 2, 3, 4, 5 ]
            return arr; //[ 1, 2, 3, 4, 5 ]
          }

> 改变原来的Set结构

        changeSet(){
             let set = new Set([1, 2, 3]);
             //方法二
             set = new Set([...set].map((val) => val * 2));
        
             //方法二
             set = new Set(Array.from(set, val => val * 2)); 
             return set; // [ 4, 8, 12 ]
        }

##### 2. Map
Map结构提供了“值-值”对应,而Object结构提供了“字符串-值”对应,Map结构是一种更加完善的Hash结构实现

> 设置值set(),获取值get(),判断存在has(),删除delete(),清空用clear(),多次对同一个键赋值,后面覆盖前面

        testMap(){
              const m = new Map();
              const o = {p:'hello uct'};
        
              m.set(o,'content');
              m.set(o,'new content')
              console.log(m.get(o)); // new content
              console.log(m.has(o)); // true
              m.delete(o);
              console.log(m.has(o)); // false
          }

> Map 和 对象互相转换

        //Map转为对象
        strMapToObj(strMap){//Map to Object 
            let obj = Object.create(null);
            for(let [k,v] of strMap){
                obj[k] = v;
            }
            return obj;
        }
        //对象转为Map
        objToStrMap(obj){// Object to Map
            let strMap = new Map();
            for(let k of Object.keys(obj)){
                strMap.set(k,obj[k]);
            }
            return strMap;
        }

##### 3. WeakMap
WeakMap只接受对象作为键名(Null除外)

> WeakMap 应用的典型场景就是以DOM节点作为键名的场景

        const listener = new WeakMap();

        listener.set(element1,handler1);
        listener.set(element2,handler2);

        element1.addEventListener('click',listener.get(element1),false);
        element2.addEventListener('click',listener.get(element2),false);

        //一旦DOM对象消失,与它绑定的监听函数也会自动消失

> 部署私有属性

        const _counter = new WeakMap();
        const _action = new WeakMap();
        
        class Countdown {
            constructor(counter, action) {
                _counter.set(this, counter);//weakmap的私有属性
                _action.set(this, action);  //weakmap的私有属性
            }
            dec() {
                let counter = _counter.get(this);
                if (counter < 1) return;
                counter--;
                console.log(counter);
                _counter.set(this, counter);
                if (counter === 0) {
                    _action.get(this)();
                }
            }
        }
        
        export default Countdown;

### 结语
*本来觉得没有必要写这个笔记,但是,看和写真的是两回事。书中的例子大都是抽象化的,也就是为了让大家看起来直观,去掉了上下文,有些还有错。自己动手写的话(运用实例),印象会非常深刻,在真实开发环境中会想到去用它。笔记的下篇已经在写了,相对上篇,我更期待下篇,NSL...*

### 参考链接



[ECMAScript 6 入门](http://es6.ruanyifeng.com/)

[UNDERSTANDING ES6 英文原版](https://leanpub.com/understandinges6/read)

标签:

版权申明:本站文章部分自网络,如有侵权,请联系:west999com@outlook.com
特别注意:本站所有转载文章言论不代表本站观点,本站所提供的摄影照片,插画,设计作品,如需使用,请与原作者联系,版权归原作者所有

上一篇:ES6要点总结(4)-箭头函数=&gt;

下一篇:Angular组件——组件生命周期(二)