Vue组件间常用的通信方式总结

本文总结了在vue-cli创建的vue项目中,三种常用的组件间通信方式。包括父组件通过Prop向子组件传递数据,子组件使用$emit向父组件发送消息,以及使用Vuex让多个组件共享状态实现通讯,还提及了各方式的使用方法和注意事项。

在使用 vue-cli 创建的 vue 项目中进行组件间的通信,以下是三种常用的组件间通信的方式总结。

1、父组件向子组件传递数据----------------------------> 通过Prop

2、子组件向父组件发送消息---------------------------->使用 $emit

3、组件间与组件间通信----------------------------------->使用Vuex

 

父组件向子组件传递数据

$ vue init webpack vue-demo

首先通过Vue-cli命令创建一个webpack模板项目,为项目简单起见,这里在Install vue-router、insatll ESLint、install unit test都选择NO。

 删除掉src/components目录下面的HelloWorld.vue组件,在该目录下新建一个Child.vue组件,现在的目录结构长这样。

准备工作完毕,接下来看一下 父组件利用props往子组件传输数据。

修改Child.vue组件:

用一个 props 选项将要传递的属性包含在该组件可接受的 prop 列表中,当一个值传递给一个 prop 特性的时候,它就变成了那个组件实例的一个属性,这里则传递了一个msg属性。

<template>
  <div class="mychild">
    <h2>子组件</h2>
    <p>{{msg}}</p>
  </div>
</template>

<script>
  export default{
    props:["msg"],
  }
</script>

<style>
.mychild
{
  border:1px solid black;
  width:400px;
}
</style>

修改App.vue组件:

这里使用了 v-bind 来动态传递 prop,也就是说当parentMsg改变的时候,子组件对应的值也会改变。这里添加了一个按钮来改变parentMsg值,用于观察。

<template>
  <div id="app">
      <button v-on:click="changeMsg">改变msg内容</button>
      <child  v-bind:msg="parentMsg"></child>
  </div>
</template>

<script>
import Child from './components/Child'

export default {
  name: 'App',
  data(){
    return {
        parentMsg:'original parent message'
    }
  },
  components: {
    Child
  },
  methods:{
    changeMsg:function(){
        console.log('1');
        this.parentMsg='parent message has changed!'
    }
  }
}
</script>

<style>

</style>

启动项目 :

$ npm run dev

 运行可以看到父组件向子组件传递了一个数据,点击改点msg内容按钮,对应子组件的值也改变了。

注意:在利用props实现传值的过程中理论上是要实现单向传递,即父组件改变相关参数的值,子组件也相应变化,但是子组件对参数的改变不应该影响父组件。但是当props中接收的是父组件传递的引用类型(对象或者是数组)时,在子组件中对数据改变时,父组件中的数据也会相应的改变,因为两者是指向的同一地址内存。如果不想子组件的改变影响父组件可以利用深拷贝,将接受的数据进行深拷贝后在子组件中使用,而不直接操作接受的数据。深拷贝可以直接利用ES6中的obj=Object。assign({},myMessage)(在computed中定义),这样子组件的改动将不会影响到父组件。

 

子组件向父组件发送消息

Vue 实例提供了一个自定义事件的系统来解决这个问题。我们可以调用内建的 $emit 方法并传入事件的名字,来向父级组件触发一个事件,然后我们可以用 v-on 在父组件上监听这个事件,就像监听一个原生 DOM 事件一样。

 

同样用上面的这个项目,修改child.vue组件:

在点击button的时候,会使用$emit向父组件派发一个事件

<template>
  <div class="mychild">
    <h2>子组件</h2>
    <button v-on:click="sendMsgToParent">向父组件传值</button>
  </div>
</template>

<script>
  export default{
    methods:{
      sendMsgToParent:function(){
        this.$emit('myevent',"some message from child!");
      }
    }
  }
</script>

<style>
.mychild
{
  border:1px solid black;
  width:400px;
}
</style>

修改App.vue:

在父级组件监听myevent这个事件,在事件处理函数中,值将会作为第一个参数传入这个方法。以此获取到子组件向父组件发送的消息。

<template>
  <div id="app">
       <div>
         msg:{{msgFromChild}}
       </div>
      <child  v-on:myevent="getMsgFromChild"></child>
  </div>
</template>

<script>
import Child from './components/Child'

export default {
  name: 'App',
  data(){
      return{
          msgFromChild:''
      }
  },
  components: {
    Child
  },
  methods:{
    getMsgFromChild:function(data){
        this.msgFromChild=data;
    }
  }
}
</script>

<style>

</style>

启动项目 :

$ npm run dev

 运行后点击按钮可以看到利用事件机制子组件向父组件传递了一个参数

注意: 当有组件嵌套时则需要利用该机制一层一层的触发到指定层,不然直接在顶层监听子组件的子组件的事件是监听不到的,需要先向父组件派发,父组件在向上层触发

 

使用Vuex让多个vue组件中共享状态实现组件通讯 

Vuex是一个专为Vue开发的应用程序的状态管理模式,它采用集中式存储管理应用的所有组件的状态,并以相应的规则保证状态以一种可预测的方式发生变化。

Vuex特点:

  • 状态存储是响应式:当 Vue 组件从 store 中读取状态的时候,若 store 中的状态发生变化,那么相应的组件也会相应地得到高效更新。
  • 不能随意修改Vuex的状态:改变 store 中的状态的唯一途径就是显式地提交 (commit) mutation。

 

同样是上面这个项目,我们首先要安装Vuex

$ npm install vuex --save

接下来在src目录下面创建一个store.js文件,在src/components下面创建OtherChild.vue文件。现在的目录结构长这样:

修改main.js,引入store

// The Vue build version to load with the `import` command
// (runtime-only or standalone) has been set in webpack.base.conf with an alias.
import Vue from 'vue'
import App from './App'
import store from './store'

Vue.config.productionTip = false

/* eslint-disable no-new */
new Vue({
  store,
  el: '#app',
  components: { App },
  template: '<App/>'
})

store.js:

import Vue from 'vue'
import Vuex from 'vuex'

Vue.use(Vuex)

export default new Vuex.Store({
  state: {
    someMsg:'some msg'
  },
  mutations: {
    change_msg(state, data)
    {
      state.someMsg=data;
    }
  }
})

App.vue:

<template>
  <div id="app">
      <Child></Child>
      <OtherChild></OtherChild>
  </div>
</template>

<script>
import Child from './components/Child';
import OtherChild from './components/OtherChild.vue';



export default {
  name: 'App',
  components: {
    Child,
    OtherChild
  }
}
</script>

<style>

</style>

Child.vue:

<template>
  <div class="mychild">
    <h3>子组件1</h3>
    <div>
      {{msg}}
    </div>
    <button v-on:click="changeState">子组件1改变状态</button>
  </div>
</template>

<script>
  export default{
    computed: {
      // 返回store中的值
      msg() {
        return this.$store.state.someMsg;
      }
    },
    methods:{
      changeState:function(){
          console.log('1');
        this.$store.commit('change_msg','子组件1改变了msg');
      }
    }
  }
</script>

<style>
.mychild
{
  border:1px solid black;
  width:400px;
}
</style>

 OtherChild.vue:

<template>
  <div class="mychild">
    <h3>子组件2</h3>
    <div>
      {{msg}}
    </div>
    <button v-on:click="changeState">子组件2  改变状态</button>
  </div>
</template>

<script>
  export default{
    computed: {
      // 返回store中的值
      msg() {
        return this.$store.state.someMsg;
      }
    },
    methods:{
      changeState:function(){
        this.$store.commit('change_msg','子组件2 改变了msg');
      }
    }
  }
</script>

<style>
  .mychild
  {
    border:1px solid black;
    width:400px;
  }
</style>

启动项目 :

$ npm run dev

 运行后分别点击组件1下面的button和组件2下面的button。可以看到使用vuex让组件间的状态进行了集中的管理,而且Vuex 的状态存储是响应式的。当 Vue 组件从 store 中读取状态的时候,若 store 中的状态发生变化,那么相应的组件也会相应地得到高效更新。

<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、付费专栏及课程。

余额充值