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

相关推荐
GISer_Jing29 分钟前
React过渡更新:优化渲染性能的秘密
javascript·react.js·ecmascript
烛阴1 小时前
带你用TS彻底搞懂ECS架构模式
前端·javascript·typescript
wayhome在哪1 小时前
3 分钟上手!用 WebAssembly 优化前端图片处理性能(附完整代码)
javascript·性能优化·webassembly
卓码软件测评2 小时前
【第三方网站运行环境测试:服务器配置(如Nginx/Apache)的WEB安全测试重点】
运维·服务器·前端·网络协议·nginx·web安全·apache
龙在天2 小时前
前端不求人系列 之 一条命令自动部署项目
前端
开开心心就好2 小时前
PDF转长图工具,一键多页转图片
java·服务器·前端·数据库·人工智能·pdf·推荐算法
国家不保护废物2 小时前
10万条数据插入页面:从性能优化到虚拟列表的终极方案
前端·面试·性能优化
文心快码BaiduComate2 小时前
七夕,画个动态星空送给Ta
前端·后端·程序员
web前端1232 小时前
# 多行文本溢出实现方法
前端·javascript
文心快码BaiduComate2 小时前
早期人类奴役AI实录:用Comate Zulu 10min做一款Chrome插件
前端·后端·程序员