Vue单向数据流下双向绑定父子组件数据

单向数据流

Vue遵循单向数据流原则,父组件向子组件传递props,子组件不能更改props,否则会报错,也即数据只能由父组件流向子组件,子组件不能修改父组件数据。

双向数据绑定

那么在一些场景下,如弹窗场景,子组件的确需要更新父组件数据,要怎么处理呢?Vue推荐是父组件使用props传递数据给子组件,子组件再通过自定义事件通知父组件修改数据,这样一来数据就始终是由父组件控制的了。

xml 复制代码
<!-- 父组件 -->
<template>
  <button @click="show = true">打开</button>
  <Dialog
    :show="show"
    @update:show="e => show = e"
  >
  </Dialog>
</template>
<script>
  export default{
    data(){
      return {
        show: false
      }
    }
  }
</script>
xml 复制代码
<!-- 子组件 -->
<template>
  <div v-if="show">
    <button @click="$emit('update:show', false)">关闭</button>
  </div>
</template>
<script>
  export default {
    props: {
      show: {
        type: Boolean,
        default: false
      }
    }
  }
</script>

Vue2使用.sync语法糖

在Vue2中提供了修饰符.sync作为双向数据绑定的语法糖,使用.sync修饰符,父组件可以写为

ini 复制代码
<Dialog :show.sync="show"></Dialog>

子组件不变,同样通过this.$emit更新数据。

Vue3使用v-model

Vue3中则是移除了.sync修饰符,选择用v-model实现双向数据绑定,在Vue3中可以写成

ini 复制代码
<Dialog v-model:show="show"></Dialog>

子组件中同样通过emit自定义事件更新数据,注意Vue3中不能用this.$emit,而是要defineEmits(["update:show"])定义事件。

xml 复制代码
<!-- 子组件 -->
<template>
  <div v-if="show">
    <button @click="$emit('update:show', false)">关闭</button>
  </div>
</template>
<script setup>
  import { defineEmits } from "vue"
  const $emit=defineEmits(['update:show'])
</script>

defineModel

在Vue3.4+中还提供了defineModel宏,使得双向绑定的实现更加简洁,在使用子组件时直接defineModel声明一个model prop,这样,在子组件内部直接修改这个声明的值就会自动触发update事件。

如果第一个参数是一个字符串字面量,它将被用作 prop 名称;否则,prop 名称将默认为 "modelValue"。

xml 复制代码
<!-- 子组件 -->
<template>
  <div v-if="show">
    <button @click="show=false">关闭</button>
  </div>
</template>
<script setup>
 const show=defineModel('show',{type: Boolean,default: false})
</script>
相关推荐
追风筝的人er10 小时前
SpringBoot+Vue3 企业考勤如何处理法定假期?节假日方案、调休补班与工作日判断链路拆解
前端·vue.js·后端
编程老船长13 小时前
解决不同项目需要不同 Node.js 版本的问题
前端·vue.js
xiaogg367815 小时前
spring oauth2 单点登录
java·vue.js·spring
前端那点事15 小时前
Vue前端SEO优化全攻略(实操落地版,新手也能上手)
前端·vue.js
计算机学姐16 小时前
基于微信小程序的校园失物招领管理系统【uniapp+springboot+vue】
java·vue.js·spring boot·mysql·信息可视化·微信小程序·uni-app
fix一个write十个17 小时前
从零搭建音视频通话太痛苦?这个 Vue3 CallKit 让你 5 分钟搞定 1v1 + 群聊通话
前端·vue.js·github
小歪 | 前端18 小时前
VUE_运行Vue项目Network: unavailable问题解决
前端·javascript·vue.js
计算机学姐19 小时前
基于微信小程序的宠物服务系统【uniapp+springboot+vue】
java·vue.js·spring boot·mysql·微信小程序·uni-app·宠物
钱端工程师19 小时前
vue自定义一个在线查看文件的组件(.xlsx、.docx、.pdf、图片等)
javascript·vue.js·pdf
涵涵(互关)19 小时前
GoView各项目文件中的相关语法3
前端·vue.js·typescript