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 的一次全面升级。

相关推荐
阔皮大师4 小时前
INote轻量文本编辑器
java·javascript·python·c#
lbb 小魔仙4 小时前
【HarmonyOS实战】React Native 表单实战:自定义 useReactHookForm 高性能验证
javascript·react native·react.js
_codemonster4 小时前
Vue的三种使用方式对比
前端·javascript·vue.js
寻找奶酪的mouse4 小时前
30岁技术人对职业和生活的思考
前端·后端·年终总结
梦想很大很大4 小时前
使用 Go + Gin + Fx 构建工程化后端服务模板(gin-app 实践)
前端·后端·go
We་ct4 小时前
LeetCode 56. 合并区间:区间重叠问题的核心解法与代码解析
前端·算法·leetcode·typescript
张3蜂4 小时前
深入理解 Python 的 frozenset:为什么要有“不可变集合”?
前端·python·spring
无小道4 小时前
Qt——事件简单介绍
开发语言·前端·qt
广州华水科技4 小时前
GNSS与单北斗变形监测技术的应用现状分析与未来发展方向
前端
code_YuJun4 小时前
corepack 作用
前端