耀极客论坛

 找回密码
 立即注册
查看: 2957|回复: 0

proxy实现vue3数据双向绑定原理

[复制链接]

336

主题

318

帖子

22万

积分

管理员

Rank: 9Rank: 9Rank: 9

积分
220555
发表于 2022-5-9 02:22:07 | 显示全部楼层 |阅读模式
  这篇文章主要介绍了proxy实现vue3数据双向绑定原理,文章以介绍proxy的优点开始展开全文内容,围绕proxy实现vue3数据双向绑定的相关资料,,需要的朋友可以参考一下

一、proxy对比Object.defineProperty的优点
  proxy的优点:
       
  • Proxy 可以直接监听对象而非属性;   
  • Proxy 可以直接监听数组的变化;   
  • Proxy 有多达 13 种拦截方法,不限于 apply、ownKeys、deleteProperty、has 等等是Object.defineProperty 不具备的;   
  • Proxy 返回的是一个新对象,我们可以只操作新的对象达到目的,而 Object.defineProperty 只能遍历对象属性直接修改;   
  • Proxy 作为新标准将受到浏览器厂商重点持续的性能优化,也就是传说中的新标准的性能红利;
  Object.defineProperty 的优势:
  兼容性好:支持 IE9,而 Proxy 的存在浏览器兼容性问题,而且无法用 polyfill 磨平,因此 Vue 的作者才声明需要等到下个大版本( 3.0 )才能用 Proxy 重写。

二、、proxy监听对象的简单实现

1.代理对象简单实现
  1. ```javascript
  2. let data = {};// 定义一个空对象
  3. let proxy = new Proxy(data, {});// 创建一个 Proxy , 将 data 作为目标对象
  4. // 修改Proxy 代理对象的name属性
  5. proxy.name = 'shelley';
  6. console.log(proxy);
  7. console.log(data)
  8. // { name: 'shelley' }
  9. // { name: 'shelley' }
  10. ```
复制代码
2.补充知识 Reflect
  Reflect对象与Proxy对象一样,也是 ES6 为了操作对象而提供的新 API
  我们需要在 handler.set() 中 return 一个 Reflect.set(…arguments) 来进行赋值给目标对象。
       
  • Reflect.set方法设置target对象的name属性等于value。如果name属性设置了赋值函数,则赋值函数的this绑定receiver。   
  • Reflect.get方法查找并返回target对象的name属性,如果没有该属性,则返回undefined。

3.proxy方法
  handler.set()方法 属性设置操作的捕捉器。
  1. ```javascript
  2. let data = {
  3.   name: 'shelley',
  4.   age: '27'
  5. };
  6. let p = new Proxy(data, {
  7.   set(target, prop, value) {
  8.     // target = 目标对象
  9.     // prop = 设置的属性
  10.     // value = 修改后的值
  11.     console.log(target, prop, value);  // { name: 'shelley', age: '27' } age 18
  12.     return Reflect.set(...arguments);
  13.   }
  14. })
  15. p.age = 18;
  16. console.log(data);  // { name: 'shelley', age: 18 }
  17. ```
复制代码
  - handler.get() 属性读取操作的捕捉器。
  1. ```javascript
  2. let data = {
  3.   name: 'shelley',
  4.   age: 22
  5. };
  6. let p = new Proxy(data, {
  7.   get(target, prop){
  8.     console.log(target, prop);//{ name: 'shelley', age: 22 } age
  9.     return Reflect.get(...arguments);
  10.   }
  11. })
  12. console.log(p.age);//22
  13. ```
复制代码
  Object.defineProperty监听对象的简单实现
  1. ```javascript
  2. var o = {};// 创建一个新对象
  3. var bValue = 39;// 在对象中添加一个设置了存取描述符属性的示例
  4. Object.defineProperty(o, 'bValue', {  
  5.   // 这代码不会设置 o 的属性,只有访问的时候才会
  6.   get() {
  7.     return bValue;
  8.   },
  9.   set(newValue) {
  10.     console.log('set==>', newValue);
  11.     bValue = newValue;
  12.   }
  13. });
  14. console.log(o) // {}
  15. // 进入访问器代理的bValue属性的get方法,返回,并设置o对象里的bValue的值为38
  16. console.log(o.bValue); // 38
  17. // 进入访问器代理的bValue属性的set方法,设置bValue的新值,
  18. // 再进入get返回,并设置o对象里的bValue的值为40
  19. o.bValue = 40;
  20. console.log(o.bValue) // 40
  21. ```
复制代码
  小结:

       
  • es6 proxy代理器对比es5 Object.defineProperty,功能更加强大,提供了方法超多,甚至可以代理方法   
  • 为什么vue3.0才使用es6的proxy,未在2.0就使用;因为es6在部分浏览器中并未兼容,如ie的低版本,所以在**大部分主流浏览器都兼容**的情况下,才使用

三、手写vue3.0双向绑定-es6 Proxy

1、什么是Proxy
       
  • Proxy取其英文意思即“代理”。所谓代理,是你要取得某样东西或对其进行某些操作的中间媒介,而不是直接作用在这个对象上。   
  • Proxy可以理解成在目标对象前架设一层拦截层,外界访问该对象都必须先通过这层拦截,因此提供一种机制可以对外界的访问进行拦截或过滤。

2、vue.js中使用双向绑定
  1. ```javascript
  2. ‹div id="app">
  3.   ‹h2>{{msg}}‹/h2>
  4.   ‹input type="text" v-model="msg"/>
  5. ‹/div>
  6. ‹script src="https://cdn.jsdelivr.net/npm/vue/dist/vue.js">‹/script>
  7. ‹script>
  8.   let vm = new Vue({
  9.   el: '#app',
  10.   data: {
  11.     msg: 'shelley'
  12. },
  13. })
  14. ‹/script>
  15. ```
复制代码
四、Proxy对比Object.defineProperty
  Vue2.0中的双向绑定,使用Object.defineProperty()进行双向绑定
  缺点:
       
  • 无法对数组进行监听,采用的是对数组的方法进行重写(push, pop,shift,unshift等等)。对此进行双向绑定和数据监听的操作   
  • 效率差,这主要是因为对多层数据进行一次性的递归操作,如果数据很多或者是很深层次,这样性能非常的差   
  • 因为局限性,无法对新加/删除的数据进行监听,所以使用在vue2.0中使用$set进行手动添加
  - Object.definePorperty()递归遍历所有对象的所有属性,当数据层级较深时,会造成性能影响。
  - Object.definePorperty()只能作用在对象上,不能作用在数组上。
  - Object.definePorperty()只能监听定义时的属性,不能监听新增属性。
  - 由于Object.definePorperty()不能作用于数组,vue2.0选择通过重写数组方法原型的方式对数组数据进行监听,但是仍然无法监听数组索引的变化和长度的变更
  Vue3.0中双向绑定,使用Proxy和Reflect进行双向绑定
  优点:
       
  • Proxy可以对数组和对象进行拦截和监听
  缺点:
       
  • Proxy会出发多次set/get响应
  解决办法:
       
  • ①使用类似于debounce的操作,对其进行优化,使其值响应一次   
  • ②(vue3.0中的解决方式),判断key是否是target的自身属性,以及value是否和target[key]相等,可以避免多余的set/get操作
  Proxy只能代理一层,无法深度监听
       
  • ①使用深度递归,对每一层进行监听。巧妙的使用的Reflect.get()会返回对象内层结构的特性(下一层),判断下一层是否还是对象,并且使用深度递归操作。但是在性能上又很大的影响   
  • ②使用weakMap,使用两个weakMap来保存原始数据和可响应数据。访问数据时会从保存的数据中查找,如果没有再对其进行Proxy操作。
  到此这篇关于proxy实现vue3数据双向绑定原理的文章就介绍到这了,更多相关proxy实现vue3数据双向绑定 内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!


回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

Archiver|手机版|小黑屋|耀极客论坛 ( 粤ICP备2022052845号-2 )|网站地图

GMT+8, 2022-12-10 02:56 , Processed in 0.066246 second(s), 20 queries .

Powered by Discuz! X3.4

Copyright © 2001-2020, Tencent Cloud.

快速回复 返回顶部 返回列表