0 1
Proxy 简介
proxy 用于修改某些操作的默认行为,等同于在语言层面做出修改,属于一种“元编程”(meta programming),即对编程语言进行编程。
Proxy 可以理解成,在目标对象之前架设一层“拦截”,外界对该对象的访问,都必须先通过这层拦截,因此提供了一种机制,可以对外界的访问进行过滤和改写。Proxy 这个词的原意是代理,用在这里表示由它来“代理”某些操作,可以译为“代理器”。
0 2
Proxy 拦截列表
下面是 Proxy 支持的拦截操作一览,一共 13 种。
0 3
常用拦截方法
1. get 拦截对象属性的读取
let arr = [7, 8, 9];
arr = new Proxy(arr, {
get(target, prop) {
return prop in target ? target[prop] : 'error'
}
})
console.log(arr[3]) //error
console.log(arr[1]) //8
2. set 拦截对象属性的设置,返回一个布尔值
let arr = [7, 8, 9];
newArr = new Proxy(arr, {
set(target, prop, val) {
if (typeof(val) === 'number') {
target[prop] = val
return true
} else {
return false
}
}
})
newArr.push(2)
console.log(newArr[1]) //8
console.log(arr) //[7, 8, 9, 2]
newArr.push(11)
console.log(arr) //[7, 8, 9, 2, 11]
3. apply 拦截函数的调用、call和apply操作
let sum = function(...args) {
let count = 0;
args.forEach((item) => {
count += item
})
return count;
};
let p = new Proxy(sum, {
apply(target, ctx, args) {
return target(...args) * 2
}
})
console.log(p(1, 2, 3, 4)) //20
console.log(p.call(null, 1, 2, 3, 4)) //20
console.log(p.apply(null, [1, 2, 3, 4])) //20
4. construct 拦截 new 命令,返回一个对象
let User = class {
constructor(name) {
this.name = name;
}
}
User = new Proxy(User, {
construct(target, args, newTarget) {
return new target(...args)
}
})
console.log(new User('kiw')) //{name: "kiw"}
5. has 拦截propKey in proxy 的操作,返回一个布尔值
let range = {
start: 1,
end: 5,
}
range = new Proxy(range, {
has(target, prop) {
return prop >= target.start && prop <= target.end;
}
})
console.log(3 in range) //true
console.log(6 in range) //false
6. ownKeys 拦截Object.getOwnPropertyNames(proxy)、Object.getOwnPropertySymbol(proxy)、Object.keys(proxy)、 for.....in循环 返回一个数组
let userinfo = {
username: 'kiw',
age: 18,
_password: '***',
[Symbol()]: '我是唯一的',
};
let p = new Proxy(userinfo, {
ownKeys(target) {
return Object.keys(target).filter(item => {
return !item.startsWith('_')
})
}
})
console.log(Object.getOwnPropertyNames(p)) //["username", "age"]
console.log(Object.getOwnPropertySymbols(p)) //[]
console.log(Object.keys(p)) //["username", "age"]
for (const key in p) {
const element = p[key];
console.log(element) //kiw 18
}
7. deleteProperty 拦截delete proxy[propKey]的操作,返回一个布尔值
let obj = {
username: 'kiw',
age: 18,
}
let p = new Proxy(obj, {
deleteProperty(target, prop) {
delete target[prop]
return true
}
})
delete p.age;
console.log(obj) //{username:"kiw"}
更多示例, 请看阮一峰《ES6 入门教程》