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 = {}; |