Proxy
Proxy概述
Proxy
用于修改某些操作的默认行为,等同于在语言层面做出修改,所以属于一种元编程
(meta programming),即对编程语言进行编程。
Proxy
可以理解成,在目标对象之前架设一层拦截
,外界对该对象的访问,都必须先通过这层拦截,因此提供了一种机制,可以对外界的访问进行过滤和改写。Proxy
这个词的原意是代理,用在这里表示由它来代理
某些操作,可以译为代理器
。
|
|
上面代码对一个空对象架设了一层拦截,重定义了属性的读取(get)和设置(set)行为。对设置了拦截行为的对象obj,去读写它的属性,就会得到下面的结果。
|
|
上面代码说明,Proxy
实际上重载
(overload)了点运算符,即用自己的定义覆盖了语言的原始定义。
ES6原生提供Proxy构造函数,用来生成Proxy实例。
|
|
Proxy
对象的所有用法,都是上面这种形式,不同的只是handler
参数的写法。其中,new Proxy()
表示生成一个Proxy
实例,target
参数表示所要拦截的目标对象,handler
参数也是一个对象,用来定制拦截行为。下面是另一个拦截读取属性行为的例子。
|
|
上面代码中,配置对象有一个get
方法,用来拦截对目标对象属性的访问请求。get
方法的两个参数分别是目标对象和所要访问的属性。可以看到,由于拦截函数总是返回35
,所以访问任何属性都得到35
。
注意,要使得Proxy
起作用,必须针对Proxy
实例(上例是proxy对象)进行操作,而不是针对目标对象(上例是空对象)进行操作。
如果handler
没有设置任何拦截,那就等同于直接通向原对象。
|
|
上面代码中,handler
是一个空对象,没有任何拦截效果,访问handeler
就等同于访问target
。
一个技巧是将Proxy
对象,设置到object.proxy
属性,从而可以在object
对象上调用。
|
|
Proxy实例也可以作为其他对象的原型对象。
|
|
上面代码中,proxy
对象是obj
对象的原型,obj
对象本身并没有time
属性,所以根据原型链,会在proxy
对象上读取该属性,导致被拦截。同一个拦截器函数,可以设置拦截多个操作。
|
|
下面是Proxy
支持的拦截操作一览。对于可以设置、但没有设置拦截的操作,则直接落在目标对象上,按照原先的方式产生结果。
- (1)
get(target, propKey, receiver)
拦截对象属性的读取,比如proxy.foo
和proxy['foo']
,返回类型不限。最后一个参数receiver
可选,当target
对象设置了propKey
属性的get
函数时,receiver
对象会绑定get
函数的this
对象。
- (2)
set(target, propKey, value, receiver)
拦截对象属性的设置,比如proxy.foo = v或proxy[‘foo’] = v,返回一个布尔值。
- (3)
has(target, propKey)
拦截propKey in proxy的操作,以及对象的hasOwnProperty方法,返回一个布尔值。
- (4)
deleteProperty(target, propKey)
拦截delete proxy[propKey]的操作,返回一个布尔值。
- (5)
ownKeys(target)
拦截Object.getOwnPropertyNames(proxy)
、Object.getOwnPropertySymbols(proxy)
、Object.keys(proxy)
,返回一个数组。该方法返回对象所有自身的属性,而Object.keys()
仅返回对象可遍历的属性。
- (6)
getOwnPropertyDescriptor(target, propKey)
拦截Object.getOwnPropertyDescriptor(proxy, propKey)
,返回属性的描述对象。
- (7)
defineProperty(target, propKey, propDesc)
拦截Object.defineProperty(proxy, propKey, propDesc)
、Object.defineProperties(proxy, propDescs)
,返回一个布尔值。
- (8)
preventExtensions(target)
拦截Object.preventExtensions(proxy)
,返回一个布尔值。
- (9)
getPrototypeOf(target)
拦截Object.getPrototypeOf(proxy)
,返回一个对象。
- (10)
isExtensible(target)
拦截Object.isExtensible(proxy),返回一个布尔值。
- (11)
setPrototypeOf(target, proto)
拦截Object.setPrototypeOf(proxy, proto)
,返回一个布尔值。如果目标对象是函数,那么还有两种额外操作可以拦截。
- (12)
apply(target, object, args)
拦截Proxy
实例作为函数调用的操作,比如proxy(...args)
、proxy.call(object, ...args)
、proxy.apply(...)
。
- (13)
construct(target, args)
拦截Proxy
实例作为构造函数调用的操作,比如new proxy(...args)
。
Proxy实例的方法
下面是上面这些拦截方法的详细介绍。
get()
get
方法用于拦截某个属性的读取操作。上文已经有一个例子,下面是另一个拦截读取操作的例子。
|
|
上面代码表示,如果访问目标对象不存在的属性,会抛出一个错误。如果没有这个拦截函数,访问不存在的属性,只会返回undefined
。
get
方法可以继承。
|
|
上面代码中,拦截操作定义在Prototype
对象上面,所以如果读取obj
对象继承的属性时,拦截会生效。
set()
set
方法用来拦截某个属性的赋值操作。假定Person
对象有一个age
属性,该属性应该是一个不大于200
的整数,那么可以使用Proxy
保证age
的属性值符合要求。
|
|
有时,我们会在对象上面设置内部属性,属性名的第一个字符使用下划线开头,表示这些属性不应该被外部使用。结合get
和set
方法,就可以做到防止这些内部属性被外部读写。
|
|
上面代码中,只要读写的属性名的第一个字符是下划线,一律抛错,从而达到禁止读写内部属性的目的。
apply()
apply
方法拦截函数的调用、call
和apply
操作。apply
方法可以接受三个参数,分别是目标对象
、目标对象的上下文对象
(this)和目标对象的参数数组
。下面是一个例子。
|
|
上面代码中,变量p
是Proxy
的实例,当它作为函数调用时p()
,就会被apply
方法拦截,返回一个字符串。
has()
has
方法用来拦截HasProperty
操作,即判断对象是否具有某个属性时,这个方法会生效。典型的操作就是in
运算符。下面的例子使用has
方法隐藏某些属性,不被in
运算符发现。
|
|
上面代码中,如果原对象的属性名的第一个字符是下划线,proxy.has
就会返回false
,从而不会被in
运算符发现。
如果原对象不可配置或者禁止扩展,这时has
拦截会报错。
|
|
上面代码中,obj
对象禁止扩展,结果使用has
拦截就会报错。
值得注意的是,has
方法拦截的是HasProperty
操作,而不是HasOwnProperty
操作,即has
方法不判断一个属性是对象自身的属性,还是继承的属性。由于for...in
操作内部也会用到HasProperty
操作,所以has
方法在for...in
循环时也会生效。
|
|
上面代码中,for...in
循环时,has
拦截会生效,导致不符合要求的属性被排除在for...in
循环之外。
construct()
construct
方法用于拦截new
命令,下面是拦截对象的写法。
|
|
construct方法可以接受两个参数。
target
: 目标对象args
:构建函数的参数对象
|
|
construct
方法返回的必须是一个对象
,否则会报错。
|
|
deleteProperty()
deleteProperty
方法用于拦截delete
操作,如果这个方法抛出错误或者返回false
,当前属性就无法被delete
命令删除。
|
|
上面代码中,deleteProperty
方法拦截了delete
操作符,删除第一个字符为下划线的属性会报错。
defineProperty()
defineProperty
方法拦截了Object.defineProperty
操作。
|
|
上面代码中,defineProperty
方法返回false
,导致添加新属性会抛出错误。
getOwnPropertyDescriptor()
getOwnPropertyDescriptor
方法拦截Object.getOwnPropertyDescriptor
,返回一个属性描述对象或者undefined
。
|
|
上面代码中,handler.getOwnPropertyDescriptor
方法对于第一个字符为下划线的属性名会返回undefined
。
getPrototypeOf()
getPrototypeOf
方法主要用来拦截Object.getPrototypeOf()
运算符,以及其他一些操作。
Object.prototype.__proto__
Object.prototype.isPrototypeOf()
Object.getPrototypeOf()
Reflect.getPrototypeOf()
instanceof
|
|
上面代码中,getPrototypeOf
方法拦截Object.getPrototypeOf()
,返回proto
对象。
isExtensible()
isExtensible
方法拦截Object.isExtensible
操作。
|
|
上面代码设置了isExtensible
方法,在调用Object.isExtensible
时会输出called
。
这个方法有一个强限制,如果不能满足下面的条件,就会抛出错误。
|
|
下面是一个例子。
|
|
ownKeys()
ownKeys
方法用来拦截Object.keys()
操作。
|
|
上面代码拦截了对于target
对象的Object.keys()
操作,返回预先设定的数组。
下面的例子是拦截第一个字符为下划线的属性名。
|
|
preventExtensions()
preventExtensions
方法拦截Object.preventExtensions()
。该方法必须返回一个布尔值。
这个方法有一个限制,只有当Object.isExtensible(proxy)
为false
(即不可扩展)时,proxy.preventExtensions
才能返回true
,否则会报错。
|
|
上面代码中,proxy.preventExtensions
方法返回true
,但这时Object.isExtensible(proxy)
会返回true
,因此报错。
为了防止出现这个问题,通常要在proxy.preventExtensions
方法里面,调用一次Object.preventExtensions
。
|
|
setPrototypeOf()
setPrototypeOf
方法主要用来拦截Object.setPrototypeOf
方法。下面是一个例子。
|
|
上面代码中,只要修改target的原型对象,就会报错。
Proxy.revocable()
Proxy.revocable
方法返回一个可取消的Proxy
实例。
|
|
Proxy.revocable
方法返回一个对象,该对象的proxy
属性是Proxy
实例,revoke
属性是一个函数,可以取消Proxy
实例。上面代码中,当执行revoke
函数之后,再访问Proxy
实例,就会抛出一个错误。
Reflect概述
Reflect
对象与Proxy
对象一样,也是ES6
为了操作对象而提供的新API
。Reflect
对象的设计目的有这样几个。
(1)将
Object
对象的一些明显属于语言内部的方法(比如Object.defineProperty
),放到Reflect
对象上。现阶段,某些方法同时在Object
和Reflect
对象上部署,未来的新方法将只部署在Reflect
对象上。(2)修改某些
Object
方法的返回结果,让其变得更合理。比如,Object.defineProperty(obj, name, desc)
在无法定义属性时,会抛出一个错误,而Reflect.defineProperty(obj, name, desc)
则会返回false
。
|
|
- (3)让
Object
操作都变成函数行为。某些Object
操作是命令式,比如name in obj
和delete obj[name]
,而Reflect.has(obj, name)
和Reflect.deleteProperty(obj, name)
让它们变成了函数行为。
|
|
- (4)
Reflect
对象的方法与Proxy
对象的方法一一对应,只要是Proxy
对象的方法,就能在Reflect
对象上找到对应的方法。这就让Proxy
对象可以方便地调用对应的Reflect
方法,完成默认行为,作为修改行为的基础。也就是说,不管Proxy
怎么修改默认行为,你总可以在Reflect
上获取默认行为。
|
|
上面代码中,Proxy
方法拦截target
对象的属性赋值行为。它采用Reflect.set
方法将值赋值给对象的属性,然后再部署额外的功能。
下面是另一个例子。
|
|
上面代码中,每一个Proxy
对象的拦截操作get、delete、has
,内部都调用对应的Reflect
方法,保证原生行为能够正常执行。添加的工作,就是将每一个操作输出一行日志。
有了Reflect
对象以后,很多操作会更易读。
|
|
文章参考ECMAScript 6 入门