时间:2022-02-14 12:01:28 | 栏目:vue | 点击:次
proxy的优点:
Object.defineProperty 的优势:
兼容性好:支持 IE9,而 Proxy 的存在浏览器兼容性问题,而且无法用 polyfill 磨平,因此 Vue 的作者才声明需要等到下个大版本( 3.0 )才能用 Proxy 重写。
```javascript
let data = {};// 定义一个空对象
let proxy = new Proxy(data, {});// 创建一个 Proxy , 将 data 作为目标对象
// 修改Proxy 代理对象的name属性
proxy.name = 'shelley';
console.log(proxy);
console.log(data)
// { name: 'shelley' }
// { name: 'shelley' }
```
Reflect对象与Proxy对象一样,也是 ES6 为了操作对象而提供的新 API
我们需要在 handler.set() 中 return 一个 Reflect.set(…arguments) 来进行赋值给目标对象。
handler.set()方法 属性设置操作的捕捉器。
```javascript
let data = {
name: 'shelley',
age: '27'
};
let p = new Proxy(data, {
set(target, prop, value) {
// target = 目标对象
// prop = 设置的属性
// value = 修改后的值
console.log(target, prop, value); // { name: 'shelley', age: '27' } age 18
return Reflect.set(...arguments);
}
})
p.age = 18;
console.log(data); // { name: 'shelley', age: 18 }
```
- handler.get() 属性读取操作的捕捉器。
```javascript
let data = {
name: 'shelley',
age: 22
};
let p = new Proxy(data, {
get(target, prop){
console.log(target, prop);//{ name: 'shelley', age: 22 } age
return Reflect.get(...arguments);
}
})
console.log(p.age);//22
```
Object.defineProperty监听对象的简单实现
```javascript
var o = {};// 创建一个新对象
var bValue = 39;// 在对象中添加一个设置了存取描述符属性的示例
Object.defineProperty(o, 'bValue', {
// 这代码不会设置 o 的属性,只有访问的时候才会
get() {
return bValue;
},
set(newValue) {
console.log('set==>', newValue);
bValue = newValue;
}
});
console.log(o) // {}
// 进入访问器代理的bValue属性的get方法,返回,并设置o对象里的bValue的值为38
console.log(o.bValue); // 38
// 进入访问器代理的bValue属性的set方法,设置bValue的新值,
// 再进入get返回,并设置o对象里的bValue的值为40
o.bValue = 40;
console.log(o.bValue) // 40
```
小结:
es5 Object.defineProperty,功能更加强大,提供了方法超多,甚至可以代理方法
```javascript
<div id="app">
<h2>{{msg}}</h2>
<input type="text" v-model="msg"/>
</div>
<script src="https://cdn.jsdelivr.net/npm/vue/dist/vue.js"></script>
<script>
let vm = new Vue({
el: '#app',
data: {
msg: 'shelley'
},
})
</script>
```
Vue2.0中的双向绑定,使用Object.defineProperty()进行双向绑定
缺点:
- Object.definePorperty()递归遍历所有对象的所有属性,当数据层级较深时,会造成性能影响。
- Object.definePorperty()只能作用在对象上,不能作用在数组上。
- Object.definePorperty()只能监听定义时的属性,不能监听新增属性。
- 由于Object.definePorperty()不能作用于数组,vue2.0选择通过重写数组方法原型的方式对数组数据进行监听,但是仍然无法监听数组索引的变化和长度的变更
Vue3.0中双向绑定,使用Proxy和Reflect进行双向绑定
优点:
缺点:
解决办法:
debounce的操作,对其进行优化,使其值响应一次Proxy只能代理一层,无法深度监听