vue3的12种组件通信方式

embedded/2025/1/11 9:55:42/

对于日常使用vue3开发项目的前端小伙伴来说,组件通信方式可以说是必会的基本功,今天带大家一起盘下vue3的通信方式。

我们这里按照组件的关系来划分。总共包含12中组件通信方式

  • 一、父子通信
    1. props
    2. defineEmits
    3. $attrs
    4. $ref + defineExpose
    5. $parent
  • 二、兄弟组件通信
    1. mitt
    2. $parent
    3. vuex/pinia
    4. app.config.globalProperties
  • 三、跨层级通信
    1. mitt
    2. vuex/pinia
    3. provide/inject
  • 四:其它方式
    • 浏览器本地存储storage
    • 全局window对象
    • ES6模块化import/export

一:父子通信

1.1、父传子:props

最最常用的通信方式是props了,父组件通过props方式将属性传递给子组件,子组件接受props并用于数据操作和页面渲染。

注意:子组件不要直接修改父组件传递过来的props,保持自上而下单项数据流,这样会让数据的流向十分清晰,方便后续维护!

// Parent.vue
<template><Child :msg="msg"/>
</template><script setup>javascript">
import { ref } from 'vue';
import Child from './components/Child.vue';const msg = ref('hello world');
</script>
// Child.vue
<template><div>propsData:{{ msg }}</div>
</template><script setup>javascript">
defineProps({msg: String
})
</script>

1.2、子传父:defineEmits

通过defineEmits可以让子组件的值传递到父组件中。

其用法如下:

  1. 先在子组件中用defineEmits([...emitName])定义一个或多个emit,它的返回值是一个emits函数,然后可以通过调用emits函数向父组件发射时间,并携带参数。
// Child.vue
<template><div @click="onClick">child</div>
</template><script setup>javascript">
const emits = defineEmits(['update']);
const onClick = () => {emits('update', 'child update');
}
</script>
  1. 在父组件中通过@符 + 事件名监听子组件发射出来的事件,并接收其传过来的值。
// Parent.vue
<template><div>Parent<Child @update="update"/></div>
</template><script setup>javascript">
import Child from './Child.vue';
const update = val => {console.log(val); // 当子组件点击事件触发后,这里会打印 child update
}
</script>

vue2的组件中还可以通过this.$onthis.$emit监听发射事件,以达到传值的目的,但在vue3已废弃这种写法。

1.3、$attrs

如果需要在子组件中接收的props很多,如果在props声明比较繁琐,所以vue给我们提供了一个优雅的解决方案,即$attrs$attrs指的是父组件传递给子组件的所有属性中,剔除在props中定义的那部分之后,剩下的就会放在$attrs中。

举个例子:

// Parent.vue
<template><div>Parent<Child :msg1="1" :msg2="2" /></div>
</template>

这里父组件给子组件传递了两个属性msg1msg2

// Child.vue
<template><div>child: {{ $attrs }}</div>
</template><script setup>javascript">
defineProps({msg1: String
})
</script>

这里子组件使用了defineProps定义了msg1,则页面中$attrs的值为{ msg2: 2 }

还可以使用v-bind$attrs的所有数据,以属性的方式全部传递到子组件中,我们平常在封装组件的时候,这个东西就能帮助我们实现组件的属性透传,十分好用。

<template><Child v-bind="$attrs"/>
</template>

注意:在vue3中$listeners已废弃,无法使用。

1.4、$ref + defineExpose

通过$ref可以拿到组件的实例,defineExpose可以显式指定在 <script setup> 组件中要暴露出去的属性,它两一起配合使用,就能实现父子组件的通信。

其用法如下:

  1. 在子组件中通过defineExpose暴露一个update方法。
// Child.vue
<script setup>javascript">
defineExpose({update(val) {console.log('父组件传递过来的值', val);}
})
</script>
  1. 在父组件中通过ref拿到组件实例并调用子组件暴露的update方法。
// Parent.vue
<template><div>Parent<Child ref="childRef"/></div>
</template><script setup>javascript">
import Child from './Child.vue';
import { ref, onMounted } from 'vue';const childRef = ref(null);onMounted(() => {childRef.value.update('hello')
})
const update = () => {}</script>

1.5、$parent

$parent代表当前组件的父组件实例,如果当前组件是顶层组件,则$parent的值为null

我们可以通过$parent拿到父组件的实例,自然就可以进行父子组件的交互了。一般也是和defineExpose配合使用,和$ref + defineExpose用法类似,这里就不多说了。

注意:$children在vue3中已经废弃,无法使用。

二、兄弟组件

两个兄弟关系组件进行通信,我们一般会借助第三方媒介

2.1、mitt

mitt相当于我们vue2的事件总线$bus,只是vue3将其废弃,所以我们借助mitt实现类似$bus的效果。

用法如下:

  1. 安装mitt
npm install mitt
  1. 初始化mitt
// emitter.js
import mitt from'mitt';
export default mitt();

兄弟组件1:

<script setup>javascript">
import emitter from '@/utils/emitter'
emitter.on('update', (val) => {console.log('update事件触发', val)
})
</script>

兄弟组件2:

<script setup>javascript">
import emitter from '@/utils/emitter'setTimeout(() => {emitter.emit('update', 'hello')
}, 1000)
</script>

2.2、$parent

我们可以把状态(即数据)定义在父组件中,两个兄弟组件可以借助其共同的父组件共享同一份数据,间接实现通信。

2.3、vuex/pinia

vuexvue官方提供的状态管理工具,用它可以实现全局的状态共享,自然也可以实现兄弟组件的通信了。当然也可以使用pinia替代vuex

2.4、app.config.globalProperties

app.config.globalProperties是一个全局的对象,在应用内所有组件实例都能访问到,当组件属性名和它发生同名冲突时,采取就近原则,以组件的为准,这个就相当于vue2Vue.prototype

三、跨层级通信

3.1、mitt

mitt可以实现全局的通信,这个在上面介绍兄弟组件通信的时候已经说过,这里不再多说了。

3.2、vuex/pinia

vuexpinia都是全局的状态管理工具,跨层级通信也不再话下。

3.3、provide/inject

provide/injectvue3提供的可以跨层级通信的方式。

其用法如下:

  1. 父组件/根组件中定义provide提供数据
// App.vue
<script setup>javascript">
import { ref, provide } from 'vue';const name = ref('sam');
provide('name', name)
</script>
  1. 子组件/孙子组件中使用inject注入数据
// 后代组件
<script setup>javascript">
import { inject } from 'vue';const name = inject('name');
console.log('name', name.value); // 输出name
</script>

四、其它方式

4.1、浏览器本地存储storage

html5提供了一套storage API,包括了localStoragesessionStorage,它实现持久化存储、缓存等功能,自然也可以用来组件间通信了。

// 组件A
<script setup>javascript">
sessionStorage.setItem('name', 'jack');
</script>
// 组件B
<script setup>javascript">
setTimeout(() => {console.log(sessionStorage.getItem('name')); //打印 jack
}, 1000)
</script>

这里我在组件A使用sessionStarge设置了一个name值,在组件B里面就能拿到了,只要保证获取值在设置值之后执行就行了。

4.2、全局window对象(不推荐使用)

window作为一个全局对象,当然也可以使用它来通信了,不过它既然谁都可以访问到,就存在如下问题:

  1. 命令冲突问题
  2. 难以追踪数据修改,可维护性差;
  3. 挂在window上的数据难以销毁,从而造成内存泄漏

所以不推荐使用window对象进行通信。

4.3、 ES6模块化import/export

我们可以使用ES5的模块化规范import/export实现通信。

// a.js
export let a = undefined;
setTimeout(() => {a = 1;
}, 1000)
// b.js
import { a } from './a.js'setTimeout(() => {console.log(a); // 打印1
}, 2000)

由于ES module采用的是符号绑定,所以就算export的值是一个基本数据类型的值,后续修改了也能访问到。

小结

上面介绍了总结了12种vue3的组件通信方式,包括:

  1. props
  2. defineEmits
  3. $attrs
  4. $ref + defineExpose
  5. $parent
  6. mitt
  7. vuex/pinia
  8. app.config.globalProperties
  9. provide/inject
  10. 浏览器本地存储storage
  11. 全局window对象
  12. ES6模块化import/export

希望能对大家有帮助!


http://www.ppmy.cn/embedded/152974.html

相关文章

Django后端相应类设计

通用的ApiResponse类&#xff1a;用于生成统一的 API 响应格式。每个响应都包含以下字段&#xff08;每个接口最终的返回数据格式&#xff09;&#xff1a; status_code&#xff1a;HTTP 状态码&#xff08;如 200、400、500 等&#xff09;message&#xff1a;响应的描述信息…

SQL SERVER__RSN 恢复的深入解析

1. RSN 的工作原理 RSN 是 SQL Server 内部用于跟踪和管理备份和恢复操作顺序的编号。每次数据库备份&#xff08;包括完整备份、差异备份和事务日志备份&#xff09;都会生成一个唯一的 RSN。SQL Server 在恢复过程中使用 RSN 来确保备份文件按正确的顺序应用&#xff0c;从而…

万字详解 MySQL MGR 高可用集群搭建

文章目录 1、MGR 前置介绍 1.1、什么是 MGR1.2、MGR 优点1.3、MGR 缺点1.4、MGR 适用场景 2、MySQL MGR 搭建流程 2.1、环境准备2.2、搭建流程 2.2.1、配置系统环境2.2.2、安装 MySQL2.2.3、配置启动 MySQL2.2.4、修改密码、设置主从同步2.2.5、安装 MGR 插件 3、MySQL MGR 故…

【机器学习篇】探索机器学习在农业中的应用:从作物预测到精准农业

准备开启这场美妙的旅行吧&#xff01;&#xff01;&#xff01; 目录 一机器学习在农业中的重要性&#xff1a; 1.1提高产量和质量&#xff1a; 1.2资源优化配置&#xff1a; 1.3病虫害防治&#xff1a; 二作物产量预测及回归分析&#xff1a; 2.1理论基础&#xff1a…

Scala语言的面向对象编程

Scala语言的面向对象编程 面向对象编程&#xff08;Object-Oriented Programming&#xff0c;OOP&#xff09;是一种编程范式&#xff0c;它使用“对象”来组织代码&#xff0c;这些对象能够包含数据&#xff08;属性&#xff09;以及功能&#xff08;方法&#xff09;。Scala…

120.Jenkins里的Pipeline Script

目录 1. **Declarative Pipeline** 主要部分 示例 2. **Scripted Pipeline** 主要部分 示例 3. **常用指令和功能** 环境变量 工具管理 文件操作 构建触发器 并行执行 异常处理 用户交互 4.**两种类型的特点** 1. **声明式 Pipeline (Declarative Pipeline)** 中…

【前端】【HTML】入门基础知识

参考视频&#xff1a;【狂神说Java】HTML5完整教学通俗易懂_哔哩哔哩_bilibili 一、基本结构 二、基本标签 <h1>&#xff1a;一级标题&#xff0c;通常用于页面的主标题&#xff0c;字体较大且醒目。 <h2>&#xff1a;二级标题&#xff0c;用于副标题或主要章节标…

1. npm 常用命令详解

npm 常用命令详解 npm&#xff08;Node Package Manager&#xff09;是 Node.js 的包管理工具&#xff0c;用于安装和管理 Node.js 应用中的依赖库。下面是 npm 的一些常用命令及其详细解释和示例代码。 镜像源 # 查询当前使用的镜像源 npm get registry# 设置为淘宝镜像源 …