es6--let const 生成器 yield Symbol 箭头函数 SpreadJS Map/WeakMap Set/WeakSet Promise Proxy

  • let and const
  • Destructuring 解构赋值
  • Spread and Reset SpreadJS扩展语句
  • Arrow Functions 箭头函数
  • Template Literals 模板字符串 ``
  • Classes 类 继承
  • Symbols es6第七种数据类型
  • Iterators 迭代器
  • Generators 生成器
  • Promises
  • Map 与Object对比
  • WeakMaps
  • Sets
  • WeakSets
  • Proxy 代理讲解+实例
es6

es6英文网站:https://nodejs.org/en/docs/es6/

babel英文网站 (将es6转换为es5) https://babeljs.io/

一、 let 、 const

  • let 变量 只在本作用域中

    • function letTest() {
          let foo = true;
          if(true) {
              let foo = false;  //只在本级block-->{}中有效
              console.log(foo); //false
          }
          console.log(foo);  //true
      }
      letTest();
      // console.log(foo);  //未定义 报错
      
    • if else语句中var变量在条件语句外仍然有效

      function varTest() {
          var foo = true;
          if(true) {
              var foo = false;
              console.log(foo); //false 没毛病
          }
          console.log(foo);  //false 因为if里的var声明会被提到外面
      }
      varTest();
      console.log(foo); //外部未定义,报错
      
  • {里面是一块区域 block}

  • es6中避免使用var

const

  • 不能修改声明的基本数据类型的值(number string boolean undefined
    null)

    const str = 'ab';  
                str = 'bc';  
                console.log(str);// Uncaught TypeError: Assignment to constant variable. 
               //其他四种同样报错
    
  • 可以修改引用类型的引用

    //引用类型保存在内存中,js不允许直接访问内存,so在操作的时候,其实是修改的是对象的引用 
    //例如:object:
    const foo = {
    one:'hello',
    two:{
        three:'Goodbye'
        };
    };
    foo.name='world';  //增加name
    foo.two.three='not goodbye';  //修改foo.two.three里的内容
    console.log(foo);
    
    //其他引用类型 Array Function 同objext可修改增加
    

二、SpreadJS 扩展语句

https://developer.mozilla.org/zh-CN/docs/Web/JavaScript/Reference/Operators/Spread_syntax

  • 解构赋值
    var arr = [1,2,3,4,5];
    var [a,b,c,d,e] = arr;
    console.log(a);  //1
    
  • ...+数组名
let arr = [1,2,3,4];
console.log(...arr);   //1,2,3,4
```
var meats = ['bacon','ham'];
var food = ['apple',...meats,'kiwi','rice'];
console.log(food)  //[ 'apple', 'backon', 'ham', 'kiwi', 'rice' ]
```

```
function addNum(a,b,c) {
    console.log(a+b+c);
}
var num = [1,2,3];
addNum(num[0],num[1],num[2]);
addNum(...num);  //6
```

三、箭头函数 Arrow Function

  • 引入箭头函数有两个方面作用:更简短的函数 + 不绑定this

  •   var materials = [
      'Hydrogen',
      'Helium',
      'Lithium',
      'Beryllium'
    ];
    
    materials.map(function(material) { 
      return material.length;  //字符串长度
    }); // [8, 6, 7, 9]
    
    materials.map((material) => {
      return material.length;
    }); // [8, 6, 7, 9]
    
    materials.map(material => material.length); // [8, 6, 7, 9]
    
  • let circleArea3 = r=> 3.14 * r * r  
    console.log(circleArea3(7));
    
  • 讨厌的this 箭头函数出现之前

    function Person() {
      // Person() 构造函数定义 `this`作为它自己的实例.
      this.age = 0;
    
      setInterval(function growUp() {
        // 在非严格模式, growUp()函数定义 `this`作为全局对象, 
        // 与在 Person()构造函数中定义的 `this`并不相同.
        this.age++;  //这个其实等同于window.age++,和Person里的this.age没有半毛关系
      }, 1000);
    }
    
    var p = new Person();
    
  • 在ECMAScript 3/5中,通过将this值分配给封闭的变量,解决this指向问题

    function Person() {
      var that = this;
      that.age = 0;
    
      setInterval(function growUp() {
        //  回调引用的是`that`变量, 其值是预期的对象. 
        that.age++;这个就是Person里的age,不再是指windo.age的
      }, 1000);
    }
    
  • 箭头函数里不会创建自己的this,它就是使用封闭执行上下文绑定的this值,

    • so,下面代码传递给setInterval的函数(构造函数Person)内的this与封闭函数中的this值相同
    function Person(){
      this.age = 0;
    
      setInterval(() => {
        this.age++; // |this| 正确地指向person 对象
      }, 1000);
    }
    
    var p = new Person();
    

四、模板字符串直接输出变量 ${变量名}

  • 空格不折叠
let name ='apple';
 console.log(`我最爱的水果是 ${name}因为酸甜`);
 
 
 console.log("我最爱的水果是:"+ name +'因为' +
    '好吃');                 //输出不会换行,拼接字符串空格折叠
    //我最爱的水果是苹果因为好吃
 console.log(`我最爱的是 ${name}因
     为好吃`);                     //输出会换行
     //我最爱的是apple因      
     //为好吃                       

五、class类--constructor、 super 、 extends

 class Person {           //父类 
    constructor(name, age, weight) {
        this.name = name;
        this.age = age;
        this.weight = weight;
    }
    displayName() {
        console.log(this.name);
    }
    displayAge() {
        console.log(this.age);
    }
    displayWeight() {
        console.log(this.weight);
    }
}

class Programmer extends Person{   //继承父类
    constructor(name, age, weight,language) {
      super(name,age,weight);   //constructor里会先继承父类的super
      this.language = language; 
    }
    displayLanguage() {
        console.log(this.language);
    }
}
let sally = new Person('sally',21,49);
sally.displayName();
sally.displayAge();
sally.displayWeight();
console.log('---------------');
let bucky = new Programmer('bucky roberts',87,987,'javascript');
bucky.displayName();
bucky.displayAge();
bucky.displayLanguage();   //Programmer子类特有的,Person没有

六、Symbol

  • 复习:JavaScript
    • 5种基本数据类型--number string boolean undefined null +
    • 1种复杂数据类型 -- Object
  • es6添加了一种原始数据类型 : Symbol
  • Symbol()不等于任何值,不能与其他任何类型的值进行运算,
```
let sym = Symbol('foo');
console.log(sym);   //Symbol(foo)
console.log(typeof sym);   //symbol
console.log(Symbol('foo') ===Symbol('foo'));  //false
console.log(Number(3) ===Number(3));  //true
```
  • symbol不能被自动转换为字符串,尝试拼接symbol与字符串将得到TypeError错误

    var s1 = Symbol("hello");
    console.log(`输出的s1是:${s1}`);  //报错
    var s2 = s1.toString(); //或者 var s2 = String(s1); 
    console.log(`输出的s1是:${s2}`);  //输出的s1是 Symbol(hello)
    
  • 具体应用英文: http://www.zsoltnagy.eu/es6-symbols-and-its-use-cases/

七、JavaScript迭代器 函数生成器

有几个特点:

  • 1、函数名前面有一个‘*’
      2、通过调用该函数生成一个控制器
      3、调用next()方法开始执行函数
      4、碰到yield,函数会暂停
      5、再次调用next(),继续执行函数

  • 生成器是function 用来控制迭代器

  • yield 关键字:它可以暂停函数的执行,随后可以再进进入函数继续执行

  • 太抽象,举例子:

  • for循环中就是简简单单的把输出的变量一次性输出

    for (let i = 0; i < 5; i += 1) {
      console.log(i);
    }
    // 直接输出所有的  0 -> 1 -> 2 -> 3 -> 4
    
  • 现在迭代生成器函数 .next()

  • function * generatorForLoop(num) {
      for (let i = 0; i < num; i += 1) {
        yield console.log(i);
      }
    }
    
    const genForLoop = generatorForLoop(5);
    
    genForLoop.next(); // first console.log - 0
    genForLoop.next(); // 1 每调一次next,函数继续向下执行一次,
    genForLoop.next(); // 2
    genForLoop.next(); // 3
    genForLoop.next(); // 4
    

    我们每next一下,就获取代码运一步的结果,不会一次性把所有的都输出

    • 除了.next()可以迭代生成器,for循环也可以获取生成器变量的值
    function * generator(arr) {
      for (const el in arr)
        yield el;
    }
    
    const gen = generator([0, 1, 2]);
    
    for (const g of gen) {
      console.log(g); // 0 -> 1 -> 2
    }
    
    gen.next(); // {value: undefined, done: true}
    console.log(generator[0])  //undefined
    

    注意:for in循环在这里不起作用,而且变量的获取也不能通过generator[0]这种方法获得

    • 如何创建生成器
    function * generator () {}
    function* generator () {}
    function *generator () {}
    
    let generator = function * () {}
    let generator = function* () {}
    let generator = function *() {}
    
    let generator = *() => {} // SyntaxError 报错
    let generator = ()* => {} // SyntaxError 报错
    let generator = (*) => {} // SyntaxError 报错
    
    • ==上面可以看出,生成器的构建不能通过箭头函数的方式==

    下面的生成器的例子用作方法,声明方式与function一致

    class MyClass {
      *generator() {}
      * generator() {}
    }
    
    const obj = {
      *generator() {}
      * generator() {}
    }
    
    • ==生成器函数创建必须初始化==
    function * generator(arg = 'Nothing') {
      yield arg;
    }
    
    const gen0 = generator(); // 没毛病
    const gen1 = generator('Hello'); // 也没毛病
    const gen2 = new generator(); // 没有初始化,会报错
    
    generator().next(); // 这可以有用,但每次都从头开始
    

八、Yield

  • 函数中的return :意味着函数里return后面的所有都不会再执行到

    function withReturn(a) {
      let b = 5;
      return a + b;  //到这里就跳出这个函数了
      b = 6; // 我们永远都不会到这一步,再改变b
      return a * b; // 当然更不会走到这一步了
    }
    
    withReturn(6); // 11
    withReturn(6); // 11
    
    • Yield不同
    function * withYield(a) {    //生成器函数
      let b = 5;
      yield a + b;
      b = 6; // it will be re-assigned after first execution
      yield a * b;
    }
    
    const calcSix = withYield(6);
    
    calcSix.next().value; // 11
    calcSix.next().value; // 36
    
  • 解释 yeild一次返回一个值,下一次你再调同一个函数的时候,会在上一次yield值后继续进行下去

  • 在生成器中next()方法返回一个对象,这个对象包含两个属性:

    • value 和 done,value 属性表示本次 yield 表达式的返回值,done 属性为布尔类型,表示生成器后续是否还有 yield 语句,即生成器函数是否已经执行完毕并返回
    • 执行完毕即当在生成器函数中显式 return 时会导致生成器立即变为完成状态,即调用 next() 方法返回的对象的 done 为 true,
function* simpleGenerator(){
    yield 'apples';  //到这一步,停一次
    yield 'bacon';  //再停一次
    console.log('ok,this is the line after the bacon');
    yield 'corn';   //再停一次
}
let simple = simpleGenerator();
console.log(simple.next());    //{ value: 'apples', done: false }
console.log(simple.next().value);  //bacon .value只要值,
console.log(simple.next().value);  //ok,this is the line after the bacon +// corn
console.log(simple.next().value);  //结束了,没有了,再想有next就是undefined
  • 其实在生成器中,不仅yield可以使用,return也可以,只不过一旦遇到return后面的就都不会执行,直接跳出该函数了

  • function * generator() {
      yield 1;   //停一次
      return 2;   //跳出生成器函数,后面的不会执行
      yield 3; // 永远到不了这一步了
    }
    
    const gen = generator();
    
    gen.next(); // {value: 1, done: false}
    gen.next(); // {value: 2, done: true}
    gen.next(); // {value: undefined, done: true} 
    
    • yield委托
      yield * 可以在另一个生成器中起作用,这样可以一直链接无数个生成器
    function * anotherGenerator(i) {
      yield i + 1;
      yield i + 2;
      yield i + 3;
    }
    
    function * generator(i) {
    yield i;  //第一步输出 10
      yield* anotherGenerator(i);  //到anotherGenerator里 分别停11 12 13
      yield i+10;  //回到i +20 停一次 
    }
    
    var gen = generator(10);
    
    console.log(gen.next().value); // 10
    console.log(gen.next().value); // 11
    console.log(gen.next().value); // 12
    console.log(gen.next().value); // 13
    console.log(gen.next().value); // 20
    console.log(gen.next().value); // undefined
    console.log(gen.next().value); // undefined
    console.log(gen.next().value); // undefined
    console.log(gen.next().value); // undefined
    
function* getNextId(){
    let id = 0;
    while(id < 3) {
        yield id++;                 //pauses
    }
}
let createUser = getNextId();  
console.log(createUser.next().value);   //0
console.log(createUser.next().value);   //1
console.log(createUser.next().value);  //2
console.log('hello world');    
console.log(createUser.next().value);   //undefined

九、Map

  • Map 对象保存键值对。任何值(对象或者原始值) 都可以作为一个键或一个值。

  • 键值对存储,意味着每个map实例里的key都不会重复

  • 获取值通过map.get(键名) ,通过map.size获取Map里元素数量

  •    var myMap = new Map();
        console.log(typeof myMap); //object
         console.log(myMap.size);  //没有set存进去键值对,size为0
    
        myMap.set('foo','bar');
        console.log(myMap.size);  //1
         myMap.set(12,'hh');  //键值对类型不受约束
        console.log(myMap.get('foo'));   //通过get获取键值对的值
    
  • 详细了解Map和以及其与Object的区别,点击我这里

十、WeakMap

  • 不像Map有size属性获取元素数量,
  • 不能通过迭代forEach登遍历元素
  • 最大区别:如果元素删除/释放,WeakMap就将该键值对从中删除,不会再获取到,因此这也是为什么WeakMap多用于声明私有变量

Map和WeakMap的区别:

var myMap = new Map();
var myWeakMap = new WeakMap();

var obj1 = {'foo':'bar'};
var obj2 = {'bar':'baz'};
myMap.set(obj1,'hello');
myWeakMap.set(obj2,'hello');
console.log(myMap.get(obj1));  //hello
console.log(myWeakMap.get(obj2));  //hello
//存储键值对和获取的方法一致

Map WeakMap
key可以是任何数据类型 key键值是弱引用,所以必须是Object类型(除了null);值任意
Map有.size获取元素数量 没有.size属性获取数量
可用.forEach()方法遍历 没有 .foreach()方法
键值对不会自动销毁 如果WeakMap里的key被删除了,对应的值也会被立马销毁垃圾回收
  • WeakMap中,每个键对自己所引用对象的引用是 "弱引用",===》 如果没有其他引用和该键引用同一个对象,这个对象将会被当作垃圾回。

obj1 = null;
obj2 = null;
//map里元素不能主动删除释放
myMap.forEach(function (val,index) {
    console.log(index,val);   //{ foo: 'bar' } 'hello'
});

// myMap.forEach(function (val,index) {  //报错,没有该方法
//     console.log(index,val);
// });

console.log(myWeakMap.get(obj2));  //undefined  已经垃圾回收了,


myMap.delete(obj1);
myMap.forEach(function (val,index) {
    console.log(index,val);   //{ foo: 'bar' } 'hello'
});

console.log(myMap.size);   //1
console.log(myWeakMap.size);  //undefined ,没有该方法

十一、Set

  • Set对象允许你存储任何类型的唯一值,无论是原始值或者是对象引用

  • .size获取元素数量;

  • .add(某一元素值)添加元素

  • .delete(某一值)来删除该元素

  • .has(某一值)判断set中是否有该元素

  • var mySet = new Set();
    mySet.add(10);
    mySet.add('foo');
    mySet.add({'foo':'bar'});
    
    console.log(mySet.size);  //3
    mySet.delete('foo');
    console.log(mySet.size);   //2
    console.log(mySet.has(10));  //true
    
  • for遍历Set里元素

  • for(let item of mySet) {
    console.log(item);  //10 -> foo -> { foo: 'bar' }
    }
    
    for(let item of mySet.keys()) {
        console.log(item);   //10 -> foo -> { foo: 'bar' }
    }
    
    for(let item of mySet.values()) {
        console.log(item);  //10 -> foo -> { foo: 'bar' }
    }
    for(let [key,value] of mySet.entries()) {  //key和value相等
        console.log(key); //10 -> foo -> { foo: 'bar' }
           console.logvalue); //10 -> foo -> { foo: 'bar' }
    }
    
  • 与数组Array相关;存储的样式有点类似array

  • console.log(mySet);  //Set { 10, 'foo', { foo: 'bar' } }
    console.log([...mySet]);   //[ 10, 'foo', { foo: 'bar' } ]
    

十二、WeakSet

Weakset以及与Set区别

  • 仅仅是对象的集合,不像Set是任何类型的任意值

  • var mySet = new Set();
    var myWeakSet = new WeakSet();
    
    var obj1 ={};
    var obj2 = {};
    mySet.add(obj1);
    mySet.add(obj2);
    
    myWeakSet.add(obj1);
    myWeakSet.add(obj2);    //只能添加Object类型元素
    // myWeakSet.add(1);    //报错
    
  • 没有.size属性获取元素数量

  • console.log(mySet.size); //3
    console.log(myWeakSet.size);  //undefined
    
    
  • 可以通过.has判断是否有某一元素

  • console.log(mySet.has(obj1));  //true
    console.log(myWeakSet.has(obj1));  //true
    
  • 对象集合是弱引用,如果没有对应的引用,就会被垃圾回收,这也意味着就不会再存储列表里

  • console.log(mySet);  //Set { {}, {} }
    console.log(myWeakSet);  //WeakSet {}
    
  • 不能被枚举

  • for(let item of mySet) {
        console.log(item)  //{} -> {} 依然存在
    }
    // for(let item of myWeakSet) {
    //     console.log(item)  //报错, :myWeakSet is not iterable
    // }
    

十三、Proxy

计算机术语中,什么是Proxy代理:

在计算术语中,代理位于你和你正在通信的事物之间。这个术语通常应用于代理服务器,它是一个介于web浏览器(Chrome、Firefox、Safari、Edge等)和web服务器(Apache、Nginx、IIS等)之间的设备。代理服务器可以修改请求和响应。例如,它可以通过缓存定期访问属性并将其提供给多个用户来提高效率。
在ES6中proxies就是位于你的代码和object之间。它允许你执行元编程操作,比如:拦截一个呼叫来检查或变更对象的属性。
  • Proxy对象用于为基本操作定义定制行为(如属性查找、赋值、枚举、函数调用等)。
  • 看似Proxy的相关专业词汇很多,其实你只需要熟记几个关键的:
    • handler -- 实现代理行为的对象

    • target -- 代理将虚拟化的原始对象。这可能是一个JavaScript的object对象,比如jQuery库或本地对象如数组,甚至是另一个代理。

    • traps -- 处理程序中定义的函数,当调用特定的属性或方法时,它提供对目标的访问。

  • 通俗讲,Proxy代理是构造函数,返回Proxy对象,主要用于从外部控制对对象内部的访问
  • 举个例子:我们创建target对象,有以下三个属性:

  • const target = {
        a:1,
        b:2,
        c:3
    };
    
    
  • 创建handler对象拦截所有的get操作,如果target里有该属性,就返回它,没有就输出数字:42

    let handler = {
        get:function (target,name) {
             return (
                 name in target ? target[name] : 42
        )
        }
    }
    
  • 现在,我们通过传递target和handler对象来创建一个新的代理。我们的代码可以与代理交互,而不是直接访问目标对象

    const proxy = new Proxy(target,handler);
    console.log(proxy.a);  //1 proxy代理了target对象
    console.log(proxy.b);  //2
    console.log(proxy.c);  //3
    console.log(proxy.random);  //42
    
    
  • 现在我们进一步扩大代理处理机制,这样它只允许从a-z的单字符进行属性设置

  • const target = {
        a:1,
        b:2,
        c:3
    };
    const handler = {
        
        get:function (target,name) {
            return (name in target ?target[name] :42);
        },
        set:function (target, prop,value) {
            if(prop.length == 1 && prop >='a' && prop <= 'z') {
                target[prop] = value;
                return true;
            } else {
                throw new ReferenceError(prop + '不能被设置');  //这里就抛出错误,不会再往下进行
                return false;
            }
        }
    };
    const proxy = new Proxy(target,handler);
    proxy.a = 10;
    proxy.b = 20;
    proxy.ABC = 30;  //这个会报错,
    
    

Proxy的traps方法类型:

construct(target, argList) 当使用new构建新对象时
get(target, property) 该方法必须返回属性value值
set(target, property, value) 该方法用于拦截设置属性值的操作,成功时返回时true;严格模式下返回false的时候会抛出一个TypeError异常
deleteProperty(target,property) 删除对象中的属性 ,返回的必须是true或false
apply(target,thisArg,argList) has(target,property)
has(target,property) 可以看作时针对in操作的钩子, 返回值必须是true或false
ownKeys(target) 访问Object.getOwnPropertyNames(), 必须返回一个可枚举的对象
getPrototypeOf(target) 当地去代理对象的原型时调用,必须返回原型的对象或者null
setPrototypeOf(target,prototype) 设置原型对象,没有返回值
isExtensible(target) 访问Object.isExtensible(),它决定了一个对象是否能有新添加的属性,不稀罕会true或false
perventExtensions(target) 防止新属性添加到对象中,必须返回true或false
getOwnPropertyDescriptor(target,property) 它返回undefined或属性的描述:value,writable,get,set,configurable,enumerable
defineProperty(target,property,descriptor) 定义或修改对象属性,必须返回true或false。当target属性成功定义返回true,否则返回false
  • Proxies代理允许你为人和对象创建通用的包装器,而不需要改变目标对象本身的代码

eg1: 创建剖析代理

  • 用以计算属性被访问的次数。makeProfiler工厂函数,会返回Proxy对象,保留计算的状态:

  • function makeProfiler(target) {
        const count = {},
              handler = {
            get:function (target,name) {  //传的是你的对象
                if(name in target) {
                     count[name] = (count[name] || 0) + 1; //关键点
                    return target[name];  //get方法必须返回属性value值
                }
            }
              };
        return {
            proxy:new Proxy(target,handler),
            count:count
        }
    }
    
  • 现在我们可以用上面封装好的proxy外壳应用于所有的对象或者其他代理,:

    const myObject = {
        h:'hello',
        w:'world'
    };
    
  • 创建myObject代理

    const pObj = makeProfiler(myObject);
    
    console.log(pObj.proxy.h); //hello
    console.log(pObj.proxy.h);  //hello  访问了两次,count = 2;
    console.log(pObj.proxy.w);  //world
    console.log(pObj.count.h);  //2
    console.log(pObj.count.w);  //1
    

eg2:双向绑定

  • 数据绑定同步对象。当DOM发生变化时,JavaScript的MVC库通常使用它来更新内部对象,反之亦然。

    1. 创建一个input标签,id是inputName
  • <input type="text" id="'inputName" />
    
  • 创建myUser对象,

  • const myUser = {
         id:'inputname',
         name:''
     }
    
  • 当用户改变input输入值时,更新myUser里的name,通过onchange事件处理程序来实现

  • inputChange(myUser);
    
    function inputChange(myObject) {
        if (!myObject || !myObject.id) return;  //如果传入的对象不存在或者没有id属性,return出去
        const input = document.getElementById(myObject.id);
        input.addEventListener('onchange', function(e) {
            myObject.name = input.value;
        });
    }
    
  • 在js代码中修改myUse.name时,跟新input里的内容,这一步略微复杂,但是proxy代理提供了一个解决方案

  •   const inputHandler = {
         set:function (target,prop,newValue) {
             if(prop =='name' && target.id ) { //prop指的是传入的对象里的属性:
                 //更新对象的属性
                target[prop] = newValue;
                document.getElementById(target.id).value = newValue;
                return true;  //set方法返回值只有false或true;
             }
             else return false;
         }
        };
    
    
  • 创建Proxy

  •  const myUserProxy = new Proxy(myUser,inputHandler);
    
  • 手动设置新name

  •     myUserProxy.name = 'Lily';
        console.log(myUserProxy.name);  //Lily
        console.log(document.getElementById('inputName').value);  //Lily
    
  • 这不一定是最有效的数据绑定选项,但是
    Proxy代理可以允许你在不更改其他代码的情况下改变好多现有对象的行为

  • 点击链接查看: 如何通过Proxy 结合原生js/或jQuery实现类似vue的数据双向绑定

目前Proxy的力量不一定是那么显而易见,但是它们提供了强大的元编程机会,JavaScript的创造者--Brendan Eich就认为,Proxy很赞!
  • 目前,Proxy在node和各大浏览器都可以实现,除了IE11以外,但也要注意,不是所有的浏览器支持traps,查Proxy浏览器支持度:


    Proxy内建方法的各大浏览器支持度
  • 坏消息: 目前不能通过转换工具如Babel将es6的Proxy编译成es5代码,谁让proxies这么强大呢!~

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,125评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,293评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 162,054评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,077评论 1 291
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,096评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,062评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,988评论 3 417
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,817评论 0 273
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,266评论 1 310
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,486评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,646评论 1 347
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,375评论 5 342
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,974评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,621评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,796评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,642评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,538评论 2 352

推荐阅读更多精彩内容