当前位置:网站首页>JS高级 之 Proxy-Reflect 使用详解
JS高级 之 Proxy-Reflect 使用详解
2022-08-03 02:06:00 【玄鱼殇】
目录
一、以前监听对象 Object.defineProperty() - JavaScript | MDN
二、Proxy Proxy - JavaScript | MDN
三、Reflect Reflect - JavaScript | MDN
一、以前监听对象 Object.defineProperty() - JavaScript | MDN
1. 使用Object.defineProperty
const obj = {
name: 'star',
age: 14
};
for (const [key, value] of Object.entries(obj)) {
let _value = value;
// 监听
Object.defineProperty(obj, key, {
enumerable: true,
set(value) {
console.log('set:', value);
// 通过闭包使用外部的值, obj.name = value => 这样会死循环,注意
_value = value;
},
get() {
console.log('get:', _value);
// 通过闭包使用外部的值, obj.name = value => 这样会死循环,注意
return _value;
}
});
}
// get监听到
console.log(obj.name);
// set监听到
obj.name = 'coder';
// get监听到
console.log(obj.name);
2. 缺点
1. Object.defineProperty设计的初衷,不是为了去监听截止一个对象中所有的属性的
2. 想监听更加丰富的操作,比如新增属性、删除属性,那么Object.defineProperty是无能为力的
二、Proxy Proxy - JavaScript | MDN
在ES6中,新增了一个Proxy类,是用于创建一个代理的
- 也就是说,如果希望监听一个对象的相关操作,可以先创建一个代理对象(Proxy对象)
- 之后对该对象的所有操作,都通过代理对象来完成,代理对象可以监听想要对原对象进行哪些操作
- 想要侦听某些具体的操作,那么就可以在handler中添加对应的捕捉器(Trap)
1. 基本使用
const obj = {
name: 'star',
age: 18,
height: 1.88
};
const objProxy = new Proxy(obj, {});
2. set和get捕获器
set函数有四个参数:
- target:目标对象(侦听的对象)
- property:将被设置的属性key
- value:新属性值
- receiver:调用的代理对象
get函数有三个参数:
- target:目标对象(侦听的对象)
- property:被获取的属性key
- receiver:调用的代理对象
const obj = {
name: 'star',
age: 18,
height: 1.88
};
// 1. 创建proxy代理对象
const objProxy = new Proxy(obj, {
// 2. 设置set捕获器
set(target, key, newValue) {
console.log(`监听到${key}设置值`);
target[key] = newValue;
},
// 3. 设置get捕获器
get(target, key) {
console.log(`监听到${key}获取值`);
return target[key];
}
});
// 触发get捕获器
console.log(objProxy.name);
// 触发set捕获器
objProxy.name = 'coder';
// 触发get捕获器
console.log(objProxy.name);
// 4. 新增属性也能被监听到
objProxy.address = '广州市';
// 触发set捕获器
console.log(objProxy.address);
// 添加成功
console.log(obj); // {name: 'coder', age: 18, height: 1.88, address: '广州市'}
2. Proxy所有捕获器
- handler.getPrototypeOf()
- Object.getPrototypeOf 方法的捕捉器,获取对象的原型
- handler.setPrototypeOf()
- Object.setPrototypeOf 方法的捕捉器,设置对象的原型
- handler.isExtensible()
- Object.isExtensible 方法的捕捉器,判断是否可以新增属性
- handler.preventExtensions()
- Object.preventExtensions 方法的捕捉器,阻止对象扩展
- handler.getOwnPropertyDescriptor()
- Object.getOwnPropertyDescriptor 方法的捕捉器,获取自己的属性描述符
- handler.defineProperty()
- Object.defineProperty 方法的捕捉器,定义自己的属性描述符
- handler.ownKeys()
- Object.getOwnPropertyNames 方法和Object.getOwnPropertySymbols 方法的捕捉器
- handler.has()
- in 操作符的捕捉器, 用in操作符判断的时候
- handler.get()
- 属性读取操作的捕捉器
- handler.set()
- 属性设置操作的捕捉器
- handler.deleteProperty()
- delete 操作符的捕捉器
- handler.apply()
- 函数调用操作的捕捉器,函数调用apply时触发
- handler.construct()
- new 操作符的捕捉器,函数被new时触发
三、Reflect Reflect - JavaScript | MDN
Reflect也是ES6新增的一个API,是一个对象,意思是反射
1. 作用
- 主要提供了很多操作JavaScript对象的方法,有点像Object中操作对象的方法
- Reflect.getPrototypeOf(target) 类似于 Object.getPrototypeOf()
- Reflect.defineProperty(target, propertyKey, attributes) 类似于Object.defineProperty()
2. 出现的理由
比较 Reflect 和 Object 方法 - JavaScript | MDN
- 是因为在早期的ECMA规范中没有考虑到这种对 对象本身 的操作如何设计会更加规范,所以将这些API放到了Object上面
- 但是Object作为一个构造函数,这些操作实际上放到它身上并不合适
- 同时Object作为所有类的父类,导致Object非常的臃肿
- 另外还包含一些类似in、delete的操作符,有点奇怪
- ----------------------------------------------
- 所以ES6新增了Reflect,让对对象的操作全部集中到了Reflect对象上
- Reflect同时配合Proxy,可以做到不操作原对象
看文档~~~非常nice的东西
3. 和Object的区别
大部分操作完对象后都会返回bool值,用来告知是否操作成功
// 开启严格模式
'use strict';
const obj = {
name: 'why',
age: 18
};
Object.defineProperty(obj, 'name', {
configurable: false
});
// 用以前的方式进行操作
// 1. 不确定能否删除,如果这个对象定义过属性描述符configurable为false,那么就不能删除
// 严格模式下还会报错
delete obj.name;
// 2. 判断是否删除成功,需要这样判断,这样判断还不严谨,如果对象的隐式原型上或着之上有这个属性,那么就判断不成功
if (obj.name) {
console.log('name没有删除成功');
} else {
console.log('name删除成功');
}
// 使用Reflect
// 1. 就算配置了不可删除,但是使用这个deleteProperty不会报错而导致卡在这
// 2. 删除成功或者失败会返回bool值,代码更加严谨
if (Reflect.deleteProperty(obj, 'name')) {
console.log('name删除成功');
} else {
console.log('name没有删除成功');
}
console.log(obj);
4. Reflect的常见方法
是和Proxy是一一对应的,也是13个
- Reflect.getPrototypeOf(target)
- 类似于 Object.getPrototypeOf()
- Reflect.setPrototypeOf(target, prototype)
- 设置对象原型的函数. 返回一个 Boolean, 如果更新成功,则返回
true
- 设置对象原型的函数. 返回一个 Boolean, 如果更新成功,则返回
- Reflect.isExtensible(target)
- 类似于 Object.isExtensible()
- Reflect.preventExtensions(target)
- 类似于 Object.preventExtensions()。返回一个Boolean
- Reflect.getOwnPropertyDescriptor(target, propertyKey)
- 类似于 Object.getOwnPropertyDescriptor()。如果对象中存在该属性,则返回对应的属性描述符, 否则返回 undefined
- Reflect.defineProperty(target, propertyKey, attributes)
- 和 Object.defineProperty() 类似。如果设置成功就会返回 true
- Reflect.ownKeys(target)
- 返回一个包含所有自身属性(不包含继承属性)的数组。(类似于Object.keys(), 但不会受enumerable影响)
- Reflect.has(target, propertyKey)
- 判断一个对象是否存在某个属性,和 in 运算符 的功能完全相同
- Reflect.get(target, propertyKey[, receiver])
- 获取对象身上某个属性的值,类似于 target[name]
- Reflect.set(target, propertyKey, value[, receiver])
- 将值分配给属性的函数。返回一个Boolean,如果更新成功,则返回true
- Reflect.deleteProperty(target, propertyKey)
- 作为函数的delete操作符,相当于执行 delete target[name]
- Reflect.apply(target, thisArgument, argumentsList)
- 对一个函数进行调用操作,同时可以传入一个数组作为调用参数。和
Function.prototype.apply() 功能类似
- 对一个函数进行调用操作,同时可以传入一个数组作为调用参数。和
- Reflect.construct(target, argumentsList[, newTarget])
- 对构造函数进行 new 操作,相当于执行 new target(...args)
5. Reflect和Proxy搭配来使用
主要目的 :
- 代码更规范、不直接操作原对象
- 用Proxy监听、用Reflect执行
- 执行完后还知道是否操作成功
const obj = {
name: 'why',
age: 18
};
const objProxy = new Proxy(obj, {
// 监听设置
set(target, key, newValue) {
// 以前写法,不好,不知是否操作成功
// target[key] = newValue
// 现在,nice!
const isSuccess = Reflect.set(target, key, newValue);
if (!isSuccess) {
throw new Error(`set ${key} failure`);
}
},
// 监听获取
get(target, key) {
// 不直接获取,使用Reflect
return Reflect.get(target, key);
},
// 监听判断
has(target, key) {
console.log('has');
// 不直接判断,使用Reflect
return Reflect.has(target, key);
}
});
// 操作代理对象
objProxy.name = 'kobe'; // 触发set操作
console.log('name' in objProxy); // 触发set操作 true
console.log(obj); // {name: 'kobe', age: 18}
6. Reflect和Proxy搭配来使用
// 父类
function Person(name, age) {
this.name = name;
this.age = age;
}
// 子类
function Student(name, age) {
// 可以不使用构造函数继承 在babel中的源码中就做了类似的判断
// Person.call(this, name, age)
// 创建一个对象,该对象的隐式原型指向Student,同时使用Person的构造方法
const _this = Reflect.construct(Person, [name, age], Student);
return _this;
}
// const stu = new Student("why", 18)
const stu = new Student('coder', 18);
console.log(stu); // Student {name: 'coder', age: 18}
console.log(stu.__proto__ === Student.prototype); // true
边栏推荐
猜你喜欢
随机推荐
Get the first/last day of the current week, month, quarter in MySQL
MySQL-Explain详解
孩子坐不住就是不专注?猿辅导揭秘专注力的三大误区
能添加任意贴图超级复布局的初级智能文本提示器(超级版)
钻石基础知识介绍
mysql binlog日期解析成yyyy-MM-dd
Mysql-如何进行慢SQL查询
不想当Window的Dialog不是一个好Modal,弹窗翻身记...
Wireshark data capture and analysis of the transport layer protocol (TCP protocol)
45部署LVS-DR群集
开发JSP应用的基础知识
扩展卡尔曼滤波【转】
程序员写代码日常 | 每日趣闻
项目管理到底管的是什么?
Conversational Technology!
openCV第二篇
怎么从零编写一个 v3 版本的 chrome 浏览器插件实现 CSDN 博客网站的暗黑和明亮主题切换?
Usage of permute() function in pytorch
lombok 下的@Builder和@EqualsAndHashCode(callSuper = true)注解
leetcode:153. 寻找旋转排序数组中的最小值