vue2 和 vue3 响应式 代码实现原理

简介: 在创建响应式阶段,会递归遍历原始对象的所有属性,当对象属性较多、较深时,对效率的影响颇为严重。不仅如此,由于遍历属性仅在最开始完成,因此在这儿之后无法响应属性的新增和删除。

对于做vue的响应式,问自己的内心的几个问题如下:


1)怎么做的数据劫持,vue2和 vue3的区别?


答: vue2的数据劫持是使用是es5的 Object.defineProperty的setter 和

getter方法来对对于数据的读取和设置的, 但是这里有一个最大的缺陷是 需要遍历对象的每一个属性进行setter 和 getter,

对于大数据量,或者说是复杂的组件不友好,还有就是对象里面新增属性或者说是删除对象属性做不了数据劫持,所以vue2.0提出了 vue.s e t 和 v u e . set 和 vue.set和vue.delete ; vue3的数据劫持使用的是 Proxy(代理) 来实现的,可以对大数据量的setter 和 getter,

由于每个对象的getter 返回的都是一个代理,

这使得代理可以缓存,使得大数据也是可以的,缺点是ie中的浏览器是没有es6的proxy(IE浏览器得es6转换成es5得问题),vue3是放弃对ie的支持了

创建的项目的


2)简述vue2和vue3分别是如何实现响应式的?vue3在响应式上的提升在哪里?


vue2的响应式是使用Object.defineProperty完成的,它会对原始对象有侵入。

在创建响应式阶段,会递归遍历原始对象的所有属性,当对象属性较多、较深时,对效率的影响颇为严重。不仅如此,由于遍历属性仅在最开始完成,因此在这儿之后无法响应属性的新增和删除。


在收集依赖时,vue2采取的是构造函数的办法,构造函数是一个整体,不利于tree shaking。


vue3的响应式是使用Proxy完成的,它不会侵入原始对象,而是返回一个代理对象,通过操作代理对象完成响应式。

由于使用了代理对象,因此并不需要遍历原始对象的属性,只需在读取属性时动态的决定要不要继续返回一个代理,这种按需加载的模式可以完全无视对象属性的数量和深度,达到更高的执行效率。


由于ES6的Proxy可以代理更加底层的操作,因此对属性的新增、删除都可以完美响应。

在收集依赖时,vue3采取的是普通函数的做法,利用高效率的WeakMap实现依赖记录,这利于tree shaking,从而降低打包体积。


思路:如下:


vue2响应系统


vue2响应系统由两个核心部件组成:


  • 数据响应部件:该部件的作用是将一个对象的所有属性转换为getter和setter,当读取属性或设置属性时,可以发出通知


  • 依赖收集部件:该部件的作用是在一个函数的执行过程中,记录该函数所依赖的顶层函数,将来一旦发出一个通知,将重新执行该顶层函数


数据响应式实现


2020110322012562.png


依赖收集实现


20201103220153974.png


vue3响应系统


vue3中,将响应系统彻底的分离了出去成为了一个单独的库@vue/reactivity


在响应式系统中,为了更好的支持Tree Shaking,因此将所有的功能都进行了函数化,并且全部使用TypeSript重写


数据响应实现


20201103220212137.png


依赖收集


2020110322023279.png


代码如下:


vue2


// 判断传入的是否为对象
const isObj = obj => typeof obj === 'object' && obj !== null && !Array.isArray(obj);
// 一个订阅构造函数
function Dep() {
  // 用一个set或者数据来记录依赖关系
  this.recordsSets = new Set();
}
// 收集依赖
Dep.prototype.denpend = function () {
  if (currentExcuseFunc) {
    this.recordsSets.add(currentExcuseFunc);
  }
}
// 通知依赖执行
Dep.prototype.notify = function () {
  // 执行依赖
  this.recordsSets.forEach(fn => fn());
}
let currentExcuseFunc = null;
// 自动执行, 用于执行依赖关系
function autorun(fn) {
  // // 获取当前执行的函数
  // currentExcuseFunc = fn;
  // // 执行函数
  // fn();
  // currentExcuseFunc = null;
  // 为了 动态收集依赖,需要从新运行
  function funcWrapper(){
    currentExcuseFunc = funcWrapper;
    fn();
    currentExcuseFunc = null;
  }
  funcWrapper();
}
// 一个观察者模式
function observe(obj) {
  if (!isObj(obj)) {
    return;
  }
  // 遍历obj的多个数据
  Object.keys(obj).forEach(key => {
    let originValue = obj[key];
    observe(originValue) // 继续保持为响应式
    let dep = new Dep();
    Object.defineProperty(obj, key, {
      get() {
        // 收集依赖
        dep.denpend();
        return originValue;
      },
      set(val) {
        observe(val)
        originValue = val;
        // 发出通知去执行
        dep.notify();
      }
    })
  })
}
let obj = {
  name: 'cll',
  age: 23,
  sex: 'male',
  addr: {
    counter: 'China',
    city: 'JX'
  }
}
observe(obj);
autorun(() => {
  // console.log(obj.name, obj.addr.city, obj.addr.counter,'--------');
  if(obj.age % 2 === 0){
    console.log(obj.name);
  }else{
    console.log(obj.sex);
  }
})


vue3


// vue3的响应式原理
// 对象使用一个map来接受, key为obj, value为map, map 里面的key 为属性值, value是收集的依赖,
const targetMap = new WeakMap();
// 当前执行的函数
let currentExcuseFunc = null;
// 判断传入的是否为对象
const isObj = obj => typeof obj === 'object' && obj !== null && !Array.isArray(obj);
// 用于缓存对象的属性,如果存在直接返回,不存在直接返回一个代理,里面的结构式 key:obj, value: proxyobj
const cacheMapper = new WeakMap();
// 收集依赖,需要传入对象,和对应的属性
/**
 * 记录当前正在执行的函数,依赖哪个对象的哪个属性
 * 相当于 dep.depend
 * @param {*} target
 * @param {*} key
 */
function track(target, key) {
  // 从map中获取对象
  let objMap = targetMap.get(target);
  if (!objMap) {
    objMap = new Map();
    // 把第二层的map放入weakmap中
    targetMap.set(target, objMap);
  }
  let valueSets = objMap.get(key);
  if (!valueSets) {
    // 建立一个set来存放依赖
    valueSets = new Set();
    objMap.set(key, valueSets);
  }
  if (currentExcuseFunc) {
    valueSets.add(currentExcuseFunc);
  }
}
// 发出通知,执行依赖,需要传入对象和对应的属性
/**
 * 依次触发依赖该对象该属性的所有函数
 * 相当于 dep.notify
 * @param {*} target
 * @param {*} key
 */
function trigger(target, key) {
  // 从map中获取对象
  let objMap = targetMap.get(target);
  if (!objMap) {
    return;
  }
  let valueSets = objMap.get(key);
  if (!valueSets) {
    return;
  }
  // 通知所有相关的依赖,执行
  valueSets.forEach(fn => fn())
}
// 自动执行,用于去看哪些依赖为引用
function effect(fn) {
  function funcWrapper() {
    currentExcuseFunc = funcWrapper;
    fn();
    currentExcuseFunc = null;
  }
  funcWrapper();
}
// 观察者模式
function reviewer(obj) {
  // 通过代理来做
  if (!isObj(obj)) {
    return obj;
  }
  // 获取一个代理对象
  let getObj = cacheMapper.get(obj)
  if (getObj) {
    // 如果存在直接返回
    return getObj;
  }
  // 不存在new 一个代理
  getObj = new Proxy(obj, {
    get(target, key) {
      // 收集依赖
      track(target, key);
      // 返回的也需要一个代理, 不会无法遍历对象里面的对象
      return reviewer(Reflect.get(target, key));
    },
    set(target, key, value) {
      // 设置的值,也需要是一个代理
      Reflect.set(target, key, reviewer(value));
      trigger(target, key); // 发出通知,执行依赖
    },
    // 删除一个属性
    deleteProperty(target, key){
      Reflect.deleteProperty(target, key);
      trigger(target, key);
    }
  })
  cacheMapper.set(obj, getObj);
  return getObj;
}
let obj = reviewer({
  name: 'cll',
  age: 23,
  addr: {
    counter: 'China',
    city: 'JX'
  }
})
effect(() => {
  console.log(obj.name, obj.addr.city, obj.addr.counter, '---')
})
obj.name = 'chen' // --> chen JX China ---
obj.addr.city = '中国' // --> chen 中国 China ---
obj.sex = 'male' // 不会有打印```
源码地址: https://gitee.com/Cll12345/vue2-and-vue3.git
来源:渡一教育


相关文章
|
12天前
|
JavaScript API 数据处理
vue3使用pinia中的actions,需要调用接口的话
通过上述步骤,您可以在Vue 3中使用Pinia和actions来管理状态并调用API接口。Pinia的简洁设计使得状态管理和异步操作更加直观和易于维护。无论是安装配置、创建Store还是在组件中使用Store,都能轻松实现高效的状态管理和数据处理。
44 3
|
2月前
|
前端开发 JavaScript 测试技术
Vue3中v-model在处理自定义组件双向数据绑定时,如何避免循环引用?
Web 组件化是一种有效的开发方法,可以提高项目的质量、效率和可维护性。在实际项目中,要结合项目的具体情况,合理应用 Web 组件化的理念和技术,实现项目的成功实施和交付。通过不断地探索和实践,将 Web 组件化的优势充分发挥出来,为前端开发领域的发展做出贡献。
41 8
|
2月前
|
存储 JavaScript 数据管理
除了provide/inject,Vue3中还有哪些方式可以避免v-model的循环引用?
需要注意的是,在实际开发中,应根据具体的项目需求和组件结构来选择合适的方式来避免`v-model`的循环引用。同时,要综合考虑代码的可读性、可维护性和性能等因素,以确保系统的稳定和高效运行。
34 1
|
2月前
|
JavaScript
Vue3中使用provide/inject来避免v-model的循环引用
`provide`和`inject`是 Vue 3 中非常有用的特性,在处理一些复杂的组件间通信问题时,可以提供一种灵活的解决方案。通过合理使用它们,可以帮助我们更好地避免`v-model`的循环引用问题,提高代码的质量和可维护性。
44 1
|
6天前
|
JavaScript
vue使用iconfont图标
vue使用iconfont图标
51 1
|
16天前
|
JavaScript 关系型数据库 MySQL
基于VUE的校园二手交易平台系统设计与实现毕业设计论文模板
基于Vue的校园二手交易平台是一款专为校园用户设计的在线交易系统,提供简洁高效、安全可靠的二手商品买卖环境。平台利用Vue框架的响应式数据绑定和组件化特性,实现用户友好的界面,方便商品浏览、发布与管理。该系统采用Node.js、MySQL及B/S架构,确保稳定性和多功能模块设计,涵盖管理员和用户功能模块,促进物品循环使用,降低开销,提升环保意识,助力绿色校园文化建设。
|
2月前
|
JavaScript 前端开发 开发者
vue学习第一章
欢迎来到我的博客!我是瑞雨溪,一名热爱前端的大一学生,专注于JavaScript与Vue,正向全栈进发。博客分享Vue学习心得、命令式与声明式编程对比、列表展示及计数器案例等。关注我,持续更新中!🎉🎉🎉
48 1
vue学习第一章
|
2月前
|
JavaScript 前端开发 索引
vue学习第三章
欢迎来到瑞雨溪的博客,一名热爱JavaScript与Vue的大一学生。本文介绍了Vue中的v-bind指令,包括基本使用、动态绑定class及style等,希望能为你的前端学习之路提供帮助。持续关注,更多精彩内容即将呈现!🎉🎉🎉
34 1
|
2月前
|
缓存 JavaScript 前端开发
vue学习第四章
欢迎来到我的博客!我是瑞雨溪,一名热爱JavaScript与Vue的大一学生。本文介绍了Vue中计算属性的基本与复杂使用、setter/getter、与methods的对比及与侦听器的总结。如果你觉得有用,请关注我,将持续更新更多优质内容!🎉🎉🎉
40 1
vue学习第四章
|
2月前
|
JavaScript 前端开发 算法
vue学习第7章(循环)
欢迎来到瑞雨溪的博客,一名热爱JavaScript和Vue的大一学生。本文介绍了Vue中的v-for指令,包括遍历数组和对象、使用key以及数组的响应式方法等内容,并附有综合练习实例。关注我,将持续更新更多优质文章!🎉🎉🎉
32 1
vue学习第7章(循环)

热门文章

最新文章