Vue组件间通信方式都有哪些?

一、组件间通信的概念

开始之前,我们把组件间通信这个词进行拆分

  • 组件
  • 通信

都知道组件是vue最强大的功能之一,vue中每一个.vue我们都可以视之为一个组件

通信指的是发送者通过某种媒体以某种格式来传递信息到收信者以达到某个目的。广义上,任何信息的交通都是通信

组件间通信即指组件(.vue)通过某种方式来传递信息以达到某个目的

举个栗子

我们在使用UI框架中的table组件,可能会往table组件中传入某些数据,这个本质就形成了组件之间的通信

二、组件间通信解决了什么

在古代,人们通过驿站、飞鸽传书、烽火报警、符号、语言、眼神、触碰等方式进行信息传递,到了今天,随着科技水平的飞速发展,通信基本完全利用有线或无线电完成,相继出现了有线电话、固定电话、无线电话、手机、互联网甚至视频电话等各种通信方式

从上面这段话,我们可以看到通信的本质是信息同步,共享

回到vue中,每个组件之间的都有独自的作用域,组件间的数据是无法共享的

但实际开发工作中我们常常需要让组件之间共享数据,这也是组件通信的目的

要让它们互相之间能进行通讯,这样才能构成一个有机的完整系统

二、组件间通信的分类

组件间通信的分类可以分成以下

  • 父子组件之间的通信
  • 兄弟组件之间的通信
  • 祖孙与后代组件之间的通信
  • 非关系组件间之间的通信

关系图:

三、组件间通信的方案

整理vue中8种常规的通信方案

  1. 通过 props 传递
  2. 通过 $emit 触发自定义事件
  3. 使用 ref
  4. EventBus
  5. $parent 或 $root
  6. attrs 与 listeners
  7. Provide 与 Inject
  8. Vuex

props传递数据

  • 适用场景:父组件传递数据给子组件
  • 子组件设置props属性,定义接收父组件传递过来的参数
  • 父组件在使用子组件标签中通过字面量来传递值

Children.vue

props:{
    // 字符串形式
	name:String // 接收的类型参数
    // 对象形式
    age:{  
        type:Number, // 接收的类型为数值
        defaule:18,  // 默认值为18
      	require:true // age属性必须传递
    }
}

Father.vue组件

<Children name:"jack" age=18 />

$emit触发自定义事件

  • 适用场景:子组件传递数据给父组件

  • 子组件通过$emit触发自定义事件,$emit第二个参数为传递的数值

  • 父组件绑定监听器获取到子组件传递过来的参数

Chilfen.vue

this.$emit('add', good)

Father.vue

<Children @add="cartAdd($event)" />

ref

  • 父组件在使用子组件的时候设置ref
  • 父组件通过设置子组件ref来获取数据

父组件

<Children ref="foo" />

this.$refs.foo  // 获取子组件实例,通过子组件实例我们就能拿到对应的数据

EventBus

  • 使用场景:兄弟组件传值

  • 创建一个中央时间总线EventBus

  • 兄弟组件通过$emit触发自定义事件,$emit第二个参数为传递的数值

  • 另一个兄弟组件通过$on监听自定义事件

Bus.js

// 创建一个中央时间总线类
class Bus {
  constructor() {
    this.callbacks = {};   // 存放事件的名字
  }
  $on(name, fn) {
    this.callbacks[name] = this.callbacks[name] || [];
    this.callbacks[name].push(fn);
  }
  $emit(name, args) {
    if (this.callbacks[name]) {
      this.callbacks[name].forEach((cb) => cb(args));
    }
  }
}

// main.js
Vue.prototype.$bus = new Bus() // 将$bus挂载到vue实例的原型上
// 另一种方式
Vue.prototype.$bus = new Vue() // Vue已经实现了Bus的功能

Children1.vue

this.$bus.$emit('foo')

Children2.vue

this.$bus.$on('foo', this.handle)

$parent 或 $root

  • 通过共同祖辈$parent或者$root搭建通信侨联

兄弟组件

this.$parent.on('add',this.add)

另一个兄弟组件

this.$parent.emit('add')

$attrs 与 $listeners

  • 适用场景:祖先传递数据给子孙

  • 设置批量向下传属性$attrs和 $listeners

  • 包含了父级作用域中不作为 prop 被识别 (且获取) 的特性绑定 ( class 和 style 除外)。

  • 可以通过 v-bind="$attrs" 传⼊内部组件

// child:并未在props中声明foo
<p>{{$attrs.foo}}</p>

// parent
<HelloWorld foo="foo"/>
// 给Grandson隔代传值,communication/index.vue
<Child2 msg="lalala" @some-event="onSomeEvent"></Child2>

// Child2做展开
<Grandson v-bind="$attrs" v-on="$listeners"></Grandson>

// Grandson使⽤
<div @click="$emit('some-event', 'msg from grandson')">
{{msg}}
</div>

provide 与 inject

  • 在祖先组件定义provide属性,返回传递的值
  • 在后代组件通过inject接收组件传递过来的值

祖先组件

provide(){
    return {
        foo:'foo'
    }
}

后代组件

inject:['foo'] // 获取到祖先组件传递过来的值

vuex

  • 适用场景: 复杂关系的组件数据传递

  • Vuex作用相当于一个用来存储共享变量的容器

  • state用来存放共享变量的地方
  • getter,可以增加一个getter派生状态,(相当于store中的计算属性),用来获得共享变量的值
  • mutations用来存放修改state的方法。
  • actions也是用来存放修改state的方法,不过action是在mutations的基础上进行。常用来做一些异步操作

小结

  • 父子关系的组件数据传递选择 props 与 $emit进行传递,也可选择ref
  • 兄弟关系的组件数据传递可选择$bus,其次可以选择$parent进行传递
  • 祖先与后代组件数据传递可选择attrslisteners或者 Provide与 Inject
  • 复杂关系的组件数据传递可以通过vuex存放共享的变量

 

如果大家想学习前端方面的技术,我把我多年的经验分享给大家,还有一些学习资料,免费领取V:xxy12311007
 

 

<think>我们参考用户的问题和引用内容,总结Vue组件通信方式及其适用场景。用户要求了解常用方式及使用场景,我们结合引用内容进行归纳。引用[1]提到:props通信和$parent/$children方式,但后者有弊端,Vue不提倡。引用[2]列举了多种通信方式:props/$emit,$refs,$emit/$on,provide/inject,$attrs/$listeners。引用[3]强调组件间数据传递的重要性,并首先介绍父组件传值给子组件-props。引用[4]介绍了混入(mixins)和插槽(scopedslots)两种方式,混入用于复用逻辑,插槽用于内容分发。引用[5]展示了使用$root.$emit进行全局事件通信的示例。根据以上引用,我们整理Vue组件通信的常用方式:1.Props/$emit(父子组件通信)2.$refs(父组件调用子组件方法)3.EventBus(全局事件总线,使用$emit/$on)4.Vuex(状态管理,跨组件通信)5.provide/inject(祖先和后代组件通信)6.$attrs/$listeners(跨级组件通信)7.Mixins(逻辑复用,间接通信)8.ScopedSlots(作用域插槽,子组件向父组件传递数据)下面详细说明每种方式及其适用场景。</think>### Vue 组件通信方法及其使用场景 Vue 组件通信是构建复杂应用的核心,不同场景需选择合适方式。以下是主流方法及其适用场景(基于引用内容整理): --- #### 1. **Props / $emit** **原理**: - 父组件通过 `props` 向子组件传值 - 子组件通过 `$emit` 触发父组件事件 **代码示例**: ```vue <!-- 父组件 --> <ChildComponent :data="parentData" @update="handleUpdate" /> <!-- 子组件 --> <script> export default { props: ['data'], methods: { sendData() { this.$emit('update', newData) // 触发父组件事件 } } } </script> ``` **适用场景**: - ✅ 父子组件直接通信(最常见) - ✅ 数据流清晰简单的场景 - ❌ 深层嵌套组件(需层层传递) --- #### 2. **Event Bus(事件总线)** **原理**: - 创建全局 Vue 实例作为事件中心 - 通过 `$emit` 发送事件,`$on` 监听事件 **代码示例**: ```javascript // event-bus.js import Vue from 'vue' export const EventBus = new Vue() // 组件A(发送) EventBus.$emit('data-change', payload) // 组件B(接收) EventBus.$on('data-change', payload => { ... }) ``` **适用场景**: - ✅ 非父子组件通信(如兄弟组件) - ✅ 简单项目快速实现跨组件通信 - ❌ 大型项目(难以维护事件名)[^1] --- #### 3. **Vuex(状态管理)** **原理**: - 集中式存储管理应用状态 - 通过 `mutations` 同步修改,`actions` 异步操作 **代码示例**: ```javascript // store.js export default new Vuex.Store({ state: { count: 0 }, mutations: { INCREMENT(state) { state.count++ } }, actions: { fetchData({ commit }) { api.get().then(() => commit('INCREMENT')) } } }) // 组件中使用 this.$store.commit('INCREMENT') this.$store.dispatch('fetchData') ``` **适用场景**: - ✅ 中大型项目状态管理 - ✅ 多组件共享数据的场景(如用户信息) - ✅ 需要状态快照/时光旅行调试[^3] --- #### 4. **provide / inject** **原理**: - 祖先组件通过 `provide` 暴露数据 - 后代组件通过 `inject` 获取数据 **代码示例**: ```vue <!-- 祖先组件 --> <script> export default { provide() { return { theme: 'dark' } } } </script> <!-- 后代组件 --> <script> export default { inject: ['theme'] // 直接使用 this.theme } </script> ``` **适用场景**: - ✅ 深层嵌套组件传值(避免 prop 逐层传递) - ✅ 全局配置(如主题、语言)[^4] - ❌ 响应式数据(默认非响应式,需配合 Vue.observable) --- #### 5. **$refs / $parent / $children** **原理**: - 直接访问组件实例 **代码示例**: ```javascript // 父组件调用子组件方法 this.$refs.childComponent.methodName() // 子组件访问父组件 this.$parent.parentData = value ``` **适用场景**: - ✅ 需要直接操作 DOM 元素 - ✅ 父组件强制触发子组件方法 - ❌ 常规数据传递(破坏组件封装性)[^1] --- #### 6. **作用域插槽(Scoped Slots)** **原理**: - 子组件通过插槽暴露数据给父组件 **代码示例**: ```vue <!-- 子组件 --> <template> <slot :item="itemData"></slot> </template> <!-- 父组件 --> <ChildComponent> <template v-slot="slotProps"> {{ slotProps.item }} <!-- 使用子组件数据 --> </template> </ChildComponent> ``` **适用场景**: - ✅ 组件复用且需自定义渲染逻辑 - ✅ 数据与 UI 解耦的场景(如表单、列表)[^4] --- #### 7. **混入(Mixins)** **原理**: - 复用逻辑代码的混合模式 **代码示例**: ```javascript // mixin.js export default { methods: { sharedMethod() { console.log('复用方法') } } } // 组件使用 import mixin from './mixin' export default { mixins: [mixin] } ``` **适用场景**: - ✅ 多个组件共享相同逻辑(如日志记录) - ❌ 数据通信(主要用于逻辑复用)[^4] --- ### 通信方式选择指南 | **场景** | **推荐方式** | **原因** | |------------------------|-------------------|-----------------------------| | 父子组件简单通信 | Props / $emit | 官方推荐,数据流清晰 | | 非父子组件通信 | Vuex 或 Event Bus | 避免复杂事件链 | | 大型应用状态管理 | Vuex | 集中管理,支持调试工具 | | 跨越多层组件传值 | provide/inject | 避免 prop 逐层传递 | | 组件模板自定义渲染 | Scoped Slots | 灵活控制 UI 与数据解耦 | > ⚠️ 关键建议: > 1. 优先使用 **Props/$emit** 处理父子通信 > 2. 复杂项目必用 **Vuex** 管理共享状态 > 3. 避免滥用 `$parent/$children`(引用[1]指出其破坏封装性) > 4. 跨组件事件用 **Event Bus** 时注意事件名冲突 通过合理选择通信方式,可显著提升代码可维护性和性能。实际开发中常组合使用多种方案,例如:用 Vuex 管理核心状态 + Scoped Slots 处理 UI 定制化[^3][^4]。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值