Vue 2 响应式系统

"Vue 2 的响应式是怎么实现的?"

本文用一张流程图 + 四段源码级剖析,带你走完 Observer → Dep → Watcher → Scheduler 的完整链路。

一、为什么需要响应式

在传统命令式框架中,开发者必须手动调用 DOM API 更新视图,逻辑与渲染耦合。Vue 2 的核心目标是数据驱动------当对象或属性变化时,视图自动重绘。实现这一目标的底层机制就是响应式系统。

二、四大核心角色

1.Observer

进入组件生命周期 beforeCreate → created 之间,Vue 会递归遍历 data 中的每个属性,通过 Object.defineProperty 重写 getter/setter:

  • getter:收集依赖(记录谁在读我);
  • setter:派发更新(通知所有依赖我"我变了")。

动态增删属性无法被劫持,因此需要 vm.$set / vm.$delete 手动触发响应式。

数组则通过重写原型链方法(push、pop、splice 等)实现拦截,确保"数组内容变化"也能被观测。

2.Dep

每个响应式对象的每个属性都会对应一个 Dep 实例。它只做两件事:

  • 记录依赖(谁在读取我);
  • 触发更新(属性变化时通知所有依赖)。

当改变某个属性时,它会派发更新:那些用我的人听好了,我变了。

3.Watcher 当 render 函数执行时,Vue 不会直接运行它,而是交给一个 Watcher。Watcher 的职责:

  • 运行函数前,把自己设为全局变量;
  • 函数执行过程中,任何响应式数据的 getter 都会把当前 Watcher 记录到对应 Dep;
  • 当 Dep 触发更新时,Watcher 重新运行函数(通常是 render)。

每个组件实例至少有一个渲染 Watcher,确保数据变化 → 组件重新渲染。

4.Scheduler

如果同一轮事件循环内多次修改响应式数据,Watcher 会被多次标记为"待更新"。 Scheduler 会把这些 Watcher 放入微任务队列(通过 Promise.then 实现 nextTick),同一个 Watcher 只出现一次,从而避免不必要的重复渲染,实现批量异步更新。

三、整体数据流回顾

  1. 组件初始化 → Observer 劫持属性 → 建立 Dep 与 Watcher 关联
  2. 用户交互触发 setter → Dep 通知 Watcher → Watcher 加入 Scheduler 队列
  3. 微任务阶段统一 flush 队列 → 组件 render 重新执行 → 虚拟 DOM diff → 最小化 DOM 更新
相关推荐
敲敲了个代码1 小时前
多标签页强提醒不重复打扰:从“弹框轰炸”到“共享待处理队列”的实战
java·前端·javascript·面试·架构
不想上班只想要钱1 小时前
动态类名在 <swiper-slide 的复制项中没有起作用的解决方法
前端·vue.js
weixin_395448911 小时前
tidl_import_mul_rmfsd_psd_u8_3x480x544_bise_raw_dynamic.txt
java·服务器·前端
多多*2 小时前
图解Redis的分布式锁的历程 从单机到集群
java·开发语言·javascript·vue.js·spring·tomcat·maven
a程序小傲2 小时前
国家电网面试被问:FactoryBean与BeanFactory的区别和动态代理生成
java·linux·服务器·spring boot·spring·面试·职场和发展
Jinuss2 小时前
源码分析之React中updateContainerImpl方法更新容器
前端·react.js·前端框架
Mr Xu_3 小时前
Vue + Element Plus 实现前端导出 Excel 功能详解
前端·javascript·vue.js
前端大波3 小时前
vue3的自动化路由(unplugin-vue-router)
javascript·vue.js·自动化
美团测试工程师3 小时前
软件测试面试题总结【含答案】
软件测试·面试
仰泳之鹅3 小时前
【杂谈】使用Edge浏览器下载文件显示“Microsoft Defender SmartScreen 已阻止此不安全文件”的解决方法
前端·edge