介绍 
Proxy 和 Reflect 内置对象从 ECMAScript 2015 开始支持,允许拦截并定义自定义行为。
语法 
js
const p = new Proxy(target, { ...traps })- traps - 被代理对象上的自定义行为。包含捕获器(操作发生时所对应的处理函数)的方法列表。
 
- target - 被代理的目标对象。
 
js
const handler = {
  get(target, name, receiver) {
    // 如果对象中包含该属性,返回该属性,否则返回 not exists
    return name in target ? Reflect.get(...arguments) : 'not exists';
  },
  set(target, name, value, receiver) {
    // 如果属性为 age,判断值的类型是否为 number 和大于 0
    if (name === 'age') {
      if (!Number.isNaN(parseInt(value))) {
        target[name] = value > 0 ? value : 0;
        return true;
      } else throw 'invalid age';
    } else return Reflect.set(...arguments);
  },
};
const p = new Proxy({}, handler);
p.a = 1;
console.log(p.a, p.b); // 1, not exists
p.age = 12;
console.log(p.age); // 12
p.age = -12;
console.log(p.age); // 0
p.age = 'a';
console.log(p.age); // Uncaught invalid age使 Proxy 代理起作用,必须对 proxy 对象 进行操作,而不是原对象。
this 指向 
- Proxy代理的原对象内部的- this指向其代理对象。
js
const target = {
  m () {
    return this === proxy;
  }
};
const proxy = new Proxy(target, {});
console.log(target.m()); // false
console.log(proxy.m());  // true- handler拦截函数内部的- this,指向的是- handler对象本身。
js
const handler = {
  get() {
    console.log(this === handler);
  }
};
new Proxy({}, handler).foo; // true- 有些原生对象的内部属性,只有通过正确的 this才能获取。
js
const target = new Date();
new Proxy(target, {}).getDate(); // TypeError: this is not a Date object
const handler = {
  get(target, prop) {
    const val = target[prop];
    return typeof val === 'function' ? val.bind(target) : Reflect.get(...arguments);
  }
};
const proxy = new Proxy(target, handler);
console.log(proxy.getDate()); // 可以获取日期receiver 指向 
- get和- set捕获器函数的最后一个参数都是一个- receiver对象,指向当前代理对象。
js
const proxy = new Proxy({}, {
  get(target, key, receiver) {
    return receiver;
  }
});
proxy.getReceiver === proxy // true- 但是,代理对象作为其他对象的原型被继承时,receiver会指向被继承的对象。
js
let obj;
const proxy = new Proxy({ foo: 3 }, {
  get(target, prop, receiver) {
    console.log(receiver === proxy, receiver === obj);
    return target[prop];
  }
});
console.log(proxy.foo); // true false 3
obj = Object.create(proxy);
console.log(obj.foo); // false true 3- 在 handler拦截函数中,如果遇到target中有定义的getter,则getter中的this指向调用的target,即不会改变指向,但可以手动将其指向receiver。
js
const obj = { foo: 1 };
const proxy = new Proxy({
  foo: 3,
  get info() {
    return this.foo;
  }
}, {
  get(target, prop, receiver) {
    console.log(receiver === obj); // true
    console.log(target[prop]); // 3
    console.log(Reflect.get(target, prop)); // 3
    return Reflect.get(...arguments); // Reflect.get(target, prop, receiver)
  }
});
Object.setPrototypeOf(obj, proxy);
console.log(obj.info); // 1