文章详情

短信预约-IT技能 免费直播动态提醒

请输入下面的图形验证码

提交验证

短信预约提醒成功

详解vue3 响应式的实现原理

2024-04-02 19:55

关注

核心设计思想

除了组件化,Vue.js 另一个核心设计思想就是响应式。它的本质是当数据变化后会自动执行某个函数,映射到组件的实现就是,当数据变化后,会自动触发组件的重新渲染。响应式是 Vue.js 组件化更新渲染的一个核心机制。

Vue.js 2.x 响应式

我们先来回顾一下 Vue.js 2.x 响应式实现的部分: 它在内部通过 Object.defineProperty API 劫持数据的变化,在数据被访问的时候收集依赖,然后在数据被修改的时候通知依赖更新。

在 Vue.js 2.x 中,Watcher 就是依赖,有专门针对组件渲染的 render watcher。这里有两个流程,首先是依赖收集流程,组件在 render 的时候会访问模板中的数据,触发 getter 把 render watcher 作为依赖收集,并和数据建立联系;然后是派发通知流程,当我对这些数据修改的时候,会触发 setter,通知 render watcher 更新,进而触发了组件的重新渲染。

Object.defineProperty API 的一些缺点:

不能监听对象属性新增和删除;
初始化阶段递归执行 Object.defineProperty 带来的性能负担。

在 Vue.js 2.x 中,data 中定义的数据,Vue.js 内部在组件初始化的过程中会把它变成响应式,这是一个相对黑盒的过程,用户通常不会感知到。

Vue.js 3.x 响应式

Vue.js 3.0 为了解决 Object.defineProperty 的这些缺陷,使用 Proxy API 重写了响应式部分,并独立维护和发布整个 reactivity 库。

也就是在 Vue.js 3.0 中,是用 reactive 这个有魔力的函数,把数据变成了响应式。

reactive 内部通过 createReactiveObject 函数把 target 变成了一个响应式对象,这个函数主要做了以下几件事情:

1、函数首先判断 target 是不是数组或者对象类型,如果不是则直接返回。所以**原始数据 target 必须是对象或者数组**。
2、如果对一个已经是响应式的对象再次执行 reactive,还应该返回这个响应式对象。
3、如果对同一个原始数据多次执行 reactive ,那么会返回相同的响应式对象。
4、使用 canObserve 函数对 target 对象做一进步限制。
5、通过 Proxy API 劫持 target 对象,把它变成响应式。
6、给原始数据打个标识。

响应式的实现方式无非就是劫持数据,Vue.js 3.0 的 reactive API 就是通过 Proxy 劫持数据,而且由于 Proxy 劫持的是整个对象,所以我们可以检测到任何对对象的修改,弥补了 Object.defineProperty API 的不足。

依赖收集:get 函数

依赖收集发生在数据访问的阶段

get 函数主要做了四件事情:

1、对特殊的 key 做了代理
2、通过 Reflect.get 方法求值
3、执行 track 函数收集依赖(最核心)
4、对计算的值 res 进行判断,如果它也是数组或对象,则递归执行 reactive 把 res 变成响应式对象。

Object.defineProperty 是在初始化阶段,即定义劫持对象的时候就已经递归执行了,而 Proxy 是在对象属性被访问的时候才递归执行下一步 reactive,这其实是一种延时定义子对象响应式的实现,在性能上会有较大的提升

收集的依赖就是数据变化后执行的副作用函数

每次 track ,就是把当前激活的副作用函数 activeEffect 作为依赖,然后收集到 target 相关的 depsMap 对应 key 下的依赖集合 dep 中。

派发通知:set 函数

派发通知发生在数据更新的阶段

set 函数主要就做两件事情:

1、通过 Reflect.set 求值
2、通过 trigger 函数派发通知(最核心),并依据 key 是否存在于 target 上来确定通知类型,即新增还是修改。

trigger 函数主要做了四件事情:

1、通过 targetMap 拿到 target 对应的依赖集合 depsMap;
2、创建运行的 effects 集合;
3、根据 key 从 depsMap 中找到对应的 effect 添加到 effects 集合;
4、遍历 effects 执行相关的副作用函数。

每次 trigger 函数就是根据 target 和 key ,从 targetMap 中找到相关的所有副作用函数遍历执行一遍。

依赖收集和派发通知的过程中都提到副作用函数,依赖收集过程中我们把 activeEffect(当前激活副作用函数)作为依赖收集。

总结

其实 Vue.js 3.0 在响应式的实现思路和 Vue.js 2.x 差别并不大,主要就是 劫持数据的方式改成用 Proxy 实现 , 以及收集的依赖由 watcher 实例变成了组件副作用渲染函数 。

源码参考

由于源码太多,本文就不展示,可直接去: github.com/vuejs/core

packages/reactivity/src/baseHandlers.ts
packages/reactivity/src/effect.ts
packages/reactivity/src/reactive.ts
packages/reactivity/src/baseHandlers.ts
packages/reactivity/src/ref.ts

到此这篇关于vue3 响应式的实现原理的文章就介绍到这了,更多相关vue3 响应式原理内容请搜索编程网以前的文章或继续浏览下面的相关文章希望大家以后多多支持编程网!

阅读原文内容投诉

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

软考中级精品资料免费领

  • 历年真题答案解析
  • 备考技巧名师总结
  • 高频考点精准押题
  • 2024年上半年信息系统项目管理师第二批次真题及答案解析(完整版)

    难度     801人已做
    查看
  • 【考后总结】2024年5月26日信息系统项目管理师第2批次考情分析

    难度     348人已做
    查看
  • 【考后总结】2024年5月25日信息系统项目管理师第1批次考情分析

    难度     311人已做
    查看
  • 2024年上半年软考高项第一、二批次真题考点汇总(完整版)

    难度     432人已做
    查看
  • 2024年上半年系统架构设计师考试综合知识真题

    难度     220人已做
    查看

相关文章

发现更多好内容

猜你喜欢

AI推送时光机
位置:首页-资讯-前端开发
咦!没有更多了?去看看其它编程学习网 内容吧
首页课程
资料下载
问答资讯