Vue 2 与 Vue 3 双向绑定 (v-model) 区别详解

核心思想

v-model 都是一个语法糖,用于简化表单输入元素和组件与数据状态之间的双向同步

总结对比表

特性 Vue 2 Vue 3
底层原理 Object.defineProperty Proxy
默认绑定 :value + @input :modelValue + @update:modelValue
组件绑定数量 仅限 1 个 可多个 (通过参数区分,如 v-model:title
自定义修饰符 需手动处理,非常繁琐 内置支持 ,通过 modelModifiers prop 访问
替代方案 使用 .sync 修饰符实现多个"双向绑定" 移除了 .sync,其功能由多个 v-model 替代

一、底层实现原理不同

Vue 2: Object.defineProperty

  • 机制 :通过递归遍历数据对象,使用 gettersetter 拦截并监听每个属性。
  • 缺点
    • 无法检测 对象属性的添加或删除 (需用 Vue.set/this.$set)。
    • 无法监听****数组索引长度的变化。
  • 影响v-model 绑定的数据响应性有一定限制。

Vue 3: Proxy

  • 机制 :创建一个对象的代理,从而拦截并监听对该对象的任何操作
  • 优点
    • 可检测任何属性的变化(包括增、删)。
    • 完美监听数组的变化。
  • 影响 :为 v-model 提供了更强大、更高效的底层响应式支持。

二、在自定义组件上的使用(最大变化)

Vue 2: 单一 v-model

一个组件上只能有一个 v-model,它等价于传递 value prop 和监听 input 事件。

父组件

xml 复制代码
<ChildComponent v-model="pageTitle" />

<!-- 等价于 -->
<ChildComponent :value="pageTitle" @input="pageTitle = $event" />

子组件

xml 复制代码
<template>
  <input :value="value" @input="$emit('input', $event.target.value)" />
</template>
<script>
export default {
  props: ['value']
}
</script>

Vue 2 的替代方案 :使用 .sync 修饰符实现多个"双向绑定"

xml 复制代码
<ChildComponent :title.sync="pageTitle" />
<!-- 等价于 -->
<ChildComponent :title="pageTitle" @update:title="pageTitle = $event" />

Vue 3: 多个 v-model (核心优势)

一个组件上可以绑定多个 v-model,通过参数 来区分。默认的 v-model 使用 modelValueupdate:modelValue

父组件

ruby 复制代码
<UserName
  v-model:first-name="firstName"
  v-model:last-name="lastName"
/>

<!-- 等价于 -->
<UserName
  :first-name="firstName"
  :last-name="lastName"
  @update:first-name="firstName = $event"
  @update:last-name="lastName = $event"
/>

子组件 (使用 <script setup> )

ini 复制代码
<template>
  <input
    type="text"
    :value="firstName"
    @input="$emit('update:firstName', $event.target.value)"
  />
  <input
    type="text"
    :value="lastName"
    @input="$emit('update:lastName', $event.target.value)"
  />
</template>
<script setup>
defineProps({
  firstName: String,
  lastName: String
})

defineEmits(['update:firstName', 'update:lastName'])
</script>

三、处理自定义修饰符

Vue 3: 内置支持 (非常方便)

Vue 3 可以自动将修饰符传递给子组件。

父组件

ini 复制代码
<MyComponent v-model.capitalize="text" />

子组件

xml 复制代码
<template>
  <input :value="modelValue" @input="emitValue" />
</template>
<script setup>
const props = defineProps({
  modelValue: String,
  modelModifiers: { // 固定命名:prop名 + "Modifiers"
    default: () => ({}) // 默认为空对象
  }
})

const emit = defineEmits(['update:modelValue'])

function emitValue(e) {
  let value = e.target.value
  // 检查是否有.capitalize修饰符
  if (props.modelModifiers.capitalize) {
    value = value.charAt(0).toUpperCase() + value.slice(1)
  }
  emit('update:modelValue', value)
}
</script>

对于带参数的 v-model,修饰符对象名为 arg + "Modifiers"

例如:v-model:description.capitalize 对应的 prop 为 descriptionModifiers

Vue 2: 手动处理 (非常繁琐)

在 Vue 2 中实现类似功能需要更多步骤,包括使用 model 选项和计算属性,过程复杂。

核心结论

  1. 功能更强 :Vue 3 的 v-model 支持多个绑定,极大提升了组件的灵活性和复用性。
  2. API 更统一 :用参数化 v-model 取代了 Vue 2 中 v-model.sync 共存的混乱局面,概念更清晰。
  3. 开发更便捷内置修饰符处理让创建功能丰富的自定义输入组件变得非常简单。
  4. 基础更稳固 :基于 Proxy 的实现,响应式追踪能力更强,性能更好。

总结:Vue 3 的 v-model 在灵活性、功能性和开发体验上都是对 Vue 2 的一次全面升级。

相关推荐
Sheldon一蓑烟雨任平生10 分钟前
Vue3 依赖注入(provide 和 inject)
vue.js·inject·provide·vue3 依赖注入·跨级别组件通信
孤狼warrior1 小时前
爬虫进阶 JS逆向基础超详细,解锁加密数据
javascript·爬虫
前端炒粉1 小时前
18.矩阵置零(原地算法)
javascript·线性代数·算法·矩阵
listhi5202 小时前
利用React Hooks简化状态管理
前端·javascript·react.js
paopaokaka_luck2 小时前
基于SpringBoot+Vue的助农扶贫平台(AI问答、WebSocket实时聊天、快递物流API、协同过滤算法、Echarts图形化分析、分享链接到微博)
java·vue.js·spring boot·后端·websocket·spring
一点一木2 小时前
🚀 2025 年 10 月 GitHub 十大热门项目排行榜 🔥
前端·人工智能·github
华仔啊2 小时前
这个Vue3旋转菜单组件让项目颜值提升200%!支持多种主题,拿来即用
前端·javascript·css
非凡ghost2 小时前
Adobe Lightroom安卓版(手机调色软件)绿色版
前端·windows·adobe·智能手机·软件需求
Sheldon一蓑烟雨任平生3 小时前
Vue3 重构待办事项(主要练习组件化)
vue.js·重构·vue3·组件化练习
BestAns3 小时前
Postman 平替?这款轻量接口测试工具,本地运行 + 批量回归超实用!
前端