Object.defineProperty
使用Object.defineProperty对劫持访问器,在属性发生变化的时候会调用访问器方法。
1 | var scope = { |
Proxy
Proxy 可以理解成,在目标对象之前架设一层“拦截”,外界对该对象的访问,都必须先通过这层拦截,因此提供了一种机制,可以对外界的访问进行过滤和改写。Proxy 这个词的原意是代理,用在这里表示由它来“代理”某些操作,可以译为“代理器”。
1 | var proxy = new Proxy(target, handler); |
target参数表示所要拦截的目标对象,handler参数也是一个对象,用来定制拦截行为。
handler:
- get(target, propKey, receiver):拦截对象属性的读取,比如
proxy.foo和proxy['foo']。 - set(target, propKey, value, receiver):拦截对象属性的设置,比如
proxy.foo = v或proxy['foo'] = v,返回一个布尔值。
1 | var obj = new Proxy({}, { |
Proxy 实际上重载(overload)了点运算符,即用自己的定义覆盖了语言的原始定义。
- has(target, propKey):拦截
propKey in proxy的操作,返回一个布尔值。
1 | var p = new Proxy({name:'qinhanwen'}, { |
- deleteProperty(target, propKey):拦截
delete proxy[propKey]的操作,返回一个布尔值。 - ownKeys(target):拦截
Object.getOwnPropertyNames(proxy)、Object.getOwnPropertySymbols(proxy)、Object.keys(proxy)、for...in循环,返回一个数组。该方法返回目标对象所有自身的属性的属性名,而Object.keys()的返回结果仅包括目标对象自身的可遍历属性。 - getOwnPropertyDescriptor(target, propKey):拦截
Object.getOwnPropertyDescriptor(proxy, propKey),返回属性的描述对象。 - defineProperty(target, propKey, propDesc):拦截
Object.defineProperty(proxy, propKey, propDesc)、Object.defineProperties(proxy, propDescs),返回一个布尔值。 - preventExtensions(target):拦截
Object.preventExtensions(proxy),返回一个布尔值。 - getPrototypeOf(target):拦截
Object.getPrototypeOf(proxy),返回一个对象。
1 | var proto = {}; |
- isExtensible(target):拦截
Object.isExtensible(proxy),返回一个布尔值。 - setPrototypeOf(target, proto):拦截
Object.setPrototypeOf(proxy, proto),返回一个布尔值。如果目标对象是函数,那么还有两种额外操作可以拦截。
1 | var handler = { |
- apply(target, object, args):拦截 Proxy 实例作为函数调用的操作,比如
proxy(...args)、proxy.call(object, ...args)、proxy.apply(...)。 - construct(target, args):拦截 Proxy 实例作为构造函数调用的操作,比如
new proxy(...args)。
1 | var handler = { |
Reflect对象
Reflect对象与Proxy对象一样,也是 ES6 为了操作对象而提供的新 API,它的设计目的:
将
Object对象的一些明显属于语言内部的方法(比如Object.defineProperty),放到Reflect对象上。现阶段,某些方法同时在Object和Reflect对象上部署,未来的新方法将只部署在Reflect对象上。也就是说,从Reflect对象上可以拿到语言内部的方法。修改某些
Object方法的返回结果,让其变得更合理。比如,Object.defineProperty(obj, name, desc)在无法定义属性时,会抛出一个错误,而Reflect.defineProperty(obj, name, desc)则会返回false。让
Object操作都变成函数行为。某些Object操作是命令式,比如name in obj和delete obj[name],而Reflect.has(obj, name)和Reflect.deleteProperty(obj, name)让它们变成了函数行为。Reflect对象的方法与Proxy对象的方法一一对应,只要是Proxy对象的方法,就能在Reflect对象上找到对应的方法。这就让Proxy对象可以方便地调用对应的Reflect方法,完成默认行为,作为修改行为的基础。也就是说,不管Proxy怎么修改默认行为,你总可以在Reflect上获取默认行为。
它提供的静态方法:
- Reflect.apply(target, thisArg, args)
- Reflect.construct(target, args)
Reflect.construct方法等同于new target(...args),这提供了一种不使用new,来调用构造函数的方法。
1 | function Greeting(name) { |
- Reflect.get(target, name, receiver)
1 | var myObject = { |
如果name属性部署了读取函数(getter),则读取函数的this绑定receiver。
1 | var myObject = { |
- Reflect.set(target, name, value, receiver)
Reflect.set方法设置target对象的name属性等于value。
1 | var myObject = { |
如果name属性设置了赋值函数,则赋值函数的this绑定receiver。
1 | var myObject = { |
- Reflect.defineProperty(target, name, desc)
1 | function MyDate() { |
- Reflect.deleteProperty(target, name)
- Reflect.has(target, name)
Reflect.has方法对应name in obj里面的in运算符。
1 | var myObject = { |
- Reflect.ownKeys(target)
- Reflect.isExtensible(target)
- Reflect.preventExtensions(target)
- Reflect.getOwnPropertyDescriptor(target, name)
- Reflect.getPrototypeOf(target)
Reflect.getPrototypeOf方法用于读取对象的__proto__属性,对应Object.getPrototypeOf(obj)。
1 | const myObj = new FancyThing(); |
- Reflect.setPrototypeOf(target, prototype)
1 | const myObj = {}; |