概述:es6
Proxy 用于修改某些操做的默认行为,等同于在语言层面作出修改,因此属于一种“元编程”(meta programming),即对编程语言进行编程。
编程
Proxy 能够理解成,在目标对象以前架设一层“拦截”,外界对该对象的访问,都必须先经过这层拦截,所以提供了一种机制,能够对外界的访问进行过滤和改写。Proxy 这个词的原意是代理,用在这里表示由它来“代理”某些操做,能够译为“代理器”。app
var obj = new Proxy({}, { get: function (target, key, receiver) { console.log(`getting ${key}!`); return Reflect.get(target, key, receiver); }, set: function (target, key, value, receiver) { console.log(`setting ${key}!`); return Reflect.set(target, key, value, receiver); } });
上面代码对一个空对象架设了一层拦截,重定义了属性的读取(get
)和设置(set
)行为。这里暂时先不解释具体的语法,只看运行结果。对设置了拦截行为的对象obj
,去读写它的属性,就会获得下面的结果。编程语言
obj.count = 1 // setting count! ++obj.count // getting count! // setting count! // 2
上面代码说明,Proxy 实际上重载(overload)了点运算符,即用本身的定义覆盖了语言的原始定义。函数
ES6 原生提供 Proxy 构造函数,用来生成 Proxy 实例。this
var proxy = new Proxy(target, handler);
Proxy 对象的全部用法,都是上面这种形式,不一样的只是handler
参数的写法。其中,new Proxy()
表示生成一个Proxy
实例,target
参数表示所要拦截的目标对象,handler
参数也是一个对象,用来定制拦截行为。spa
下面是另外一个拦截读取属性行为的例子。prototype
var proxy = new Proxy({}, { get: function(target, property) { return 35; } }); proxy.time // 35 proxy.name // 35 proxy.title // 35
上面代码中,做为构造函数,Proxy
接受两个参数。第一个参数是所要代理的目标对象(上例是一个空对象),即若是没有Proxy
的介入,操做原来要访问的就是这个对象;第二个参数是一个配置对象,对于每个被代理的操做,须要提供一个对应的处理函数,该函数将拦截对应的操做。好比,上面代码中,配置对象有一个get
方法,用来拦截对目标对象属性的访问请求。get
方法的两个参数分别是目标对象和所要访问的属性。能够看到,因为拦截函数老是返回35
,因此访问任何属性都获得35
。代理
注意,要使得Proxy
起做用,必须针对Proxy
实例(上例是proxy
对象)进行操做,而不是针对目标对象(上例是空对象)进行操做。code
若是handler
没有设置任何拦截,那就等同于直接通向原对象。
var target = {}; var handler = {}; var proxy = new Proxy(target, handler); proxy.a = 'b'; target.a // "b"
上面代码中,handler
是一个空对象,没有任何拦截效果,访问proxy
就等同于访问target
。
一个技巧是将 Proxy 对象,设置到object.proxy
属性,从而能够在object
对象上调用。
var object = { proxy: new Proxy(target, handler) };
Proxy 实例也能够做为其余对象的原型对象。
var proxy = new Proxy({}, { get: function(target, property) { return 35; } }); let obj = Object.create(proxy); obj.time // 35
上面代码中,proxy
对象是obj
对象的原型,obj
对象自己并无time
属性,因此根据原型链,会在proxy
对象上读取该属性,致使被拦截。
同一个拦截器函数,能够设置拦截多个操做。
var handler = { get: function(target, name) { if (name === 'prototype') { return Object.prototype; } return 'Hello, ' + name; }, apply: function(target, thisBinding, args) { return args[0]; }, construct: function(target, args) { return {value: args[1]}; } }; var fproxy = new Proxy(function(x, y) { return x + y; }, handler); fproxy(1, 2) // 1 new fproxy(1,2) // {value: 2} fproxy.prototype === Object.prototype // true fproxy.foo // "Hello, foo"
转自: ECMAScript 6 入门
做者:阮一峰