携手创作,共同成长!这是我参与「掘金日新计划 · 8 月更文挑战」的第18天,点击查看活动详情 >>
前言
自从es6以后,JavaScript提供了Proxy和Rflect对象支持,允许你拦截并定义基本语言操作的自定义行为(例如,属性查找,赋值,枚举,函数调用等)。借助这两个对象,你可以在 JavaScript 元级别进行编程。
Proxy
Proxy 对象用于创建一个对象的代理,从而实现基本操作的拦截和自定义(如属性查找、赋值、枚举、函数调用等)。看下面这个简单的例子
let handler = {
get: function(target, name){
return name in target ? target[name] : '我爱掘金';
}};
let p = new Proxy({}, handler);
p.a = 1;
console.log(p.a, p.b); // 1, 我爱掘金
Proxy 对象定义了一个目标(这里是一个空对象)和一个实现了 get 陷阱的 handler 对象。这里,代理的对象在获取未定义的属性时不会返回 undefined,而是返回 我爱掘金。
语法
const proxy = new Proxy(target, handler)
参数
-
target要使用
Proxy包装的目标对象(可以是任何类型的对象,包括原生数组,函数,甚至另一个代理)。 -
handler一个通常以函数作为属性的对象,各属性中的函数分别定义了在执行各种操作时代理
proxy的行为。
如何创建一个可撤销的Proxy?
Proxy自身提供了一个方法revocable,通过revocable创建的proxy 是可以被撤销的
let rp = Proxy.revocable(target, handler);
创建好的rp具有量个属性,一个proxy,和正常new Proxy创建的对象几乎一样,除了可以被撤销,还有一个属性是revoke, 调用revoke可以撤销之前创建的proxy对象
一旦某个代理对象被撤销,它将变得几乎完全不可调用,在它身上执行任何的可代理操作都会抛出 TypeError异常(注意,可代理操作一共有 13种执行这 13 种操作以外的操作不会抛出异常)。一旦被撤销,这个代理对象便不可能被直接恢复到原来的状态,同时和它关联的目标对象以及处理器对象都有可能被垃圾回收掉。再次调用撤销方法 revoke() 则不会有任何效果,但也不会报错。
var revocable = Proxy.revocable({}, {
get(target, name) {
return "[[" + name + "]]";
}
});
var proxy = revocable.proxy;
proxy.foo; // "[[foo]]"
revocable.revoke();
console.log(proxy.foo); // 抛出 TypeError
proxy.foo = 1 // 还是 TypeError
delete proxy.foo; // 又是 TypeError
typeof proxy // "object",因为 typeof 不属于可代理操作
可被代理的13种操作
- handler.apply()
- handler.construct()
- handler.defineProperty()
- handler.deleteProperty()
- handler.get()
- handler.getOwnPropertyDescriptor()
- handler.getPrototypeOf()
- handler.has()
- handler.isExtensible()
- handler.ownKeys()
- handler.preventExtensions()
- handler.set()
- handler.setPrototypeOf()
下一节,将就可被代理的14种操作进行依次说明和实践