理解
在 Vue 中,MVVM(Model-View-ViewModel) 是其核心设计思想之一,它帮助实现了数据驱动的视图更新和良好的代码结构分离。我们来具体解析 Vue 是如何实现 MVVM 模式的。
🌐 MVVM 是什么?
角色 | 含义 | Vue 中的体现 |
---|---|---|
Model | 数据层 | data, computed, props 等 |
View | 视图层 | 模板(template)/ DOM |
ViewModel | 连接桥梁 | Vue 实例,本质是观察者和响应式系统 |
🔁 Vue 中 MVVM 的运作方式
Model -> View
- 数据变化后,Vue 的响应式系统会检测到变更,并自动更新 DOM。
html
<template>
<p>{{ message }}</p>
</template>
<script>
export default {
data() {
return {
message: 'Hello MVVM'
}
}
}
</script>
View -> Model
- 输入时通过 v-model 实现双向绑定,自动更新数据:
html
<input v-model="message" />
ViewModel 作为桥梁
- Vue 实例(或组件)负责监听数据变化,通知视图更新,也监听视图交互,更新数据。
✅ 为什么使用 MVVM?
- 解耦视图和逻辑:View 仅用于展示,Model 只负责存储,逻辑集中在 ViewModel 中。
- 开发体验提升:通过 v-model、v-bind、v-on 等指令,开发者只需关注数据本身,省去了操作 DOM 的繁琐。
- 便于维护与测试:业务逻辑和 DOM 分离后更好测试。
🧠 延伸理解:MVVM 与 MVC 的区别
对比点 | MVC | MVVM |
---|---|---|
控制器 Controller | 管理数据与视图之间的交互 | 被 ViewModel 替代 |
双向绑定 | 通常手动实现 | Vue 内建,自动 |
更适合 | 后端主导 | 前端交互密集的应用 |
📌 小结
- Vue 的核心机制就是基于 MVVM 架构设计。
- Vue 实例就是 MVVM 中的 ViewModel。
- 通过响应式系统和模板编译器,实现数据和视图的自动同步。
当然可以。我们可以更深入地从 原理、实现机制、以及 Vue 的源码层面 来理解 Vue 中的 MVVM 模型。下面分为几个维度逐一讲解。
🔍 MVVM 的三层角色深入解析
🧱 Model:数据模型层
• 是 Vue 实例中的 data() 返回的对象,或者组合式 API 中的 ref() 和 reactive()
• 所有状态(state)都集中在这里管理
• 响应式核心是通过 getter/setter 拦截 或 Vue 3 中的 Proxy
🧠 ViewModel:Vue 实例
• 连接 Model 和 View 的桥梁
• 包含 Vue 的响应式系统(Dep、Watcher)和 DOM 编译器(Compiler)
• 能够监听数据变化并自动更新视图(数据驱动),也能监听视图事件更新数据(事件驱动)
👁️ View:模板 / DOM
• Vue 的 template 被编译成 render function 或 VNode 虚拟 DOM 树
• 数据更新时,触发 Virtual DOM 的 diff 运算,并更新真实 DOM
⚙️ Vue 响应式系统(Vue 2 和 Vue 3)
Vue 2 响应式系统实现:
• 使用 Object.defineProperty 拦截每一个 data 属性的 get 和 set
• 利用发布-订阅模式(Dep 和 Watcher)
• Dep 是依赖收集器
• Watcher 是订阅者,负责更新视图
html
Object.defineProperty(obj, 'key', {
get() {
// 依赖收集
return value
},
set(newVal) {
// 通知视图更新
}
})
Vue 3 响应式系统实现:
• 使用 Proxy 拦截整个对象,更灵活强大
• 响应式核心库是 @vue/reactivity
• 使用 effect 和 track/trigger 实现依赖收集和更新
html
const state = reactive({ count: 0 })
🔄 双向绑定(v-model 的底层原理)
在 Vue 2 中,v-model 实际上是:
html
<input :value="message" @input="message = $event.target.value" />
在 Vue 3 中,v-model 可绑定多个属性:
html
<CustomInput v-model:title="title" />
底层通过 modelValue 和 update:modelValue 实现,增强了灵活性。
🧰 模板编译与渲染过程
1. 开发时写的 template 会被编译成 render 函数(Vue 2 编译为 VNode 树)
2. 每次数据变化,重新执行 render,得到新的 VNode
3. Vue 使用 Virtual DOM 算法进行 diff,最后以最小开销更新真实 DOM
🧪 调试 MVVM 的思维路径
- 数据改了 -> View 没变 → 检查是否是响应式的属性
- 视图事件无效 → 检查是否正确绑定、事件名是否错
- v-model 不更新 → 检查是否使用了 .lazy、.number 修饰符或未正确 emit
🖼️ 总结图示(拓展)
用户交互
↓
+--------+ 数据驱动 +---------+
| View |---------------------->| Model |
+--------+ +---------+
↑ ↓
DOM 事件 响应式更新
↑ ↓
+-------------------------------+
| ViewModel (Vue) |
+-------------------------------+