Proxy 可以理解成,在目标对象之前架设一层“拦截”,外界对该对象的访问,都必须先通过这层拦截,因此提供了一种机制,可以对外界的访问进行过滤和改写。
new Proxy()表示生成一个Proxy实例,target参数表示所要拦截的目标对象,handler参数也是一个对象,用来定制拦截行为.可省略。
上面代码中,handler是一个空对象,没有任何拦截效果,访问proxy就等同于访问target。
Proxy 实例的方法
get方法用于拦截某个属性的读取操作。
get(target, propKey, receiver)
|
|
get方法可以继承:
set(target, propKey, value, receiver)
set方法用来拦截某个属性的赋值操作
has(target, propKey)
has方法用来拦截HasProperty操作
deleteProperty(target, propKey)
deleteProperty方法用于拦截delete操作,如果这个方法抛出错误或者返回false,当前属性就无法被delete命令删除。
ownKeys(target)
ownKeys方法用来拦截对象自身属性的读取操作。具体来说,拦截以下操作:
(1)Object.getOwnPropertyNames()
(2)Object.getOwnPropertySymbols()
(3)Object.keys()
getOwnPropertyDescriptor(target, propKey)
getOwnPropertyDescriptor方法拦截Object.getOwnPropertyDescriptor(),返回一个属性描述对象或者undefined。
defineProperty(target, propKey, propDesc)
defineProperty方法拦截了Object.defineProperty操作。
preventExtensions(target)
preventExtensions方法拦截Object.preventExtensions()。该方法必须返回一个布尔值,否则会被自动转为布尔值。
这个方法有一个限制,只有目标对象不可扩展时(即Object.isExtensible(proxy)为false),proxy.preventExtensions才能返回true,否则会报错。
上面代码中,proxy.preventExtensions方法返回true,但这时Object.isExtensible(proxy)会返回true,因此报错。
为了防止出现这个问题,通常要在proxy.preventExtensions方法里面,调用一次Object.preventExtensions:
getPrototypeOf(target)
getPrototypeOf方法主要用来拦截获取对象原型。具体来说,拦截下面这些操作。
(1)Object.prototype.proto
(2)Object.prototype.isPrototypeOf()
(3)Object.getPrototypeOf()
(4)Reflect.getPrototypeOf()
(5)instanceof
isExtensible(target)
isExtensible方法拦截Object.isExtensible操作。
setPrototypeOf(target, proto)
setPrototypeOf方法主要用来拦截Object.setPrototypeOf方法。
apply(target, object, args)
apply方法拦截函数的调用、call和apply操作。
construct(target, args)
construct方法用于拦截new命令
Proxy.revocable()
Proxy.revocable方法返回一个可取消的 Proxy 实例。
Proxy.revocable方法返回一个对象,该对象的proxy属性是Proxy实例,revoke属性是一个函数,可以取消Proxy实例。上面代码中,当执行revoke函数之后,再访问Proxy实例,就会抛出一个错误。
this 问题
虽然 Proxy 可以代理针对目标对象的访问,但它不是目标对象的透明代理,即不做任何拦截的情况下,也无法保证与目标对象的行为一致。主要原因就是在 Proxy 代理的情况下,目标对象内部的this关键字会指向 Proxy 代理。