vue 组件通信方式 ,父子、隔代、兄弟 三类通信,六种方法

Vue 组件间通信只要指以下 3 类通信:父子组件通信、隔代组件通信、兄弟组件通信,下面分别介绍每种通信方式且会说明此种方法可适用于哪类组件间通信。

(1)props / $emit 适用 父子组件通信

父组件注入,子组件接收。
这种方法是 Vue 组件的基础,相信大部分同学耳闻能详,所以此处就不举例展开介绍。

注意:props是单向数据流,既只能从父级传到子级,如果想要达到双向,子级能够修改父级,则需要给props加sync修饰符。(文章后部分有详细介绍)

(2) ref$parent / $children 适用 父子组件通信

ref:如果在普通的 DOM 元素上使用,引用指向的就是 DOM 元素;如果用在子组件上,引用就指向组件实例
$parent / $children :访问父 / 子实例

(3)$attrs / $listeners 适用于 隔代组件通信

$attrs:包含了父作用域中不被 prop 所识别 (且获取) 的特性绑定 ( class 和 style 除外 )。
当一个组件没有声明任何 prop 时,这里会包含所有父作用域的绑定 ( class 和 style 除外 ),并且可以通过 v-bind="$attrs" 传入内部组件。通常配合 inheritAttrs 选项一起使用。

$listeners :包含了父作用域中的 (不含 .native 修饰器的) v-on 事件监听器。它可以通过 v-on="$listeners" 传入内部组件

(4)provide / inject 适用于 隔代组件通信

祖先组件中通过 provider 来提供变量,然后在子孙组件中通过 inject 来注入变量。provide / inject API 主要解决了跨级组件间的通信问题,不过它的使用场景,主要是子组件获取上级组件的状态,跨级组件间建立了一种主动提供与依赖注入的关系。

// 例子 全局事件中心
// App.vue
export default {
  name: "App",
  provide() {
    return {
      reload: this.loadFun,
    };
  },
 methods: {
      loadFun() {
           this.isRouterAlive = false;
           this.$nextTick(() => {
           this.isRouterAlive = true;
         });
      },
  }
}

// b.vue
export default {
  name:"xxx",
   inject: ["reload"],
  // 然后可以调用this.reload()方法
}

(5)EventBus ($emit / $on) 适用于 父子、隔代、兄弟组件通信

这种方法通过一个空的 Vue 实例作为中央事件总线(事件中心),用它来触发事件和监听事件,从而实现任何组件间的通信,包括父子、隔代、兄弟组件。

// 例子
// main.js
Vue.prototype.$EventBus = new Vue()

// a.vue   传递数据
this.$EventBus.$emit("sendMsg","发送传递数据  "+ Math.random()*10000)

// b.vue  接收数据 
this.$EventBus.$on("sendMsg",(params)=>{
      this.msg=params
})

(6)Vuex 适用于 父子、隔代、兄弟组件通信

Vuex 是一个专为 Vue.js 应用程序开发的状态管理模式。每一个 Vuex 应用的核心就是 store(仓库)。“store” 基本上就是一个容器,它包含着你的应用中大部分的状态 ( state )。

Vuex 的状态存储是响应式的。当 Vue 组件从 store 中读取状态的时候,若 store 中的状态发生变化,那么相应的组件也会相应地得到高效更新。
改变 store 中的状态的唯一途径就是显式地提交 (commit) mutation。这样使得我们可以方便地跟踪每一个状态的变化。

vuex的详细使用方法: vuex管理状态仓库使用详解


Vue 组件间通信是面试常考的知识点之一,这题有点类似于开放题,你知道的回答出越多方法越加分,表明你对 Vue 掌握的越熟练。


举例props使用

a.vue 引用了一个detail组件

// a.vue
<Detail  :goodsId="goodsId" :otherData.sync="otherData" @doSomeThing="doThing" ></Detail>
// Detail.vue
props:{
    goodsId:{
        type: String,    
    },
   otherData:{
        type: Object,
   }
}
// goodsId不能修改  
// otherData可以这样修改  
this.$emit('update:otherData', newData)   // 把父组件otherData修改成newData

// 父组件doThing方法这样调用
this.$emit('doSomeThing', params)  // 调用父组件doThing方法,并传入params参数

详解eventBus通信方法

第一步:首先需要创建事件总线并将其导出,以便其它模块可以使用或者监听它。

两个初始化事件中心的方法:

// 方法一:创建一个event-bus.js
import Vue from 'vue'
export const EventBus = new Vue()

// 方法二:main.js里直接挂载到vue原型上 
// 注意,这种方式初始化的EventBus是一个全局的事件总线。
Vue.prototype.$EventBus = new Vue()

第二步:创建了 EventBus ,接下来你需要做到的就是在你的组件中加载它,并且调用同一个方法,就如你在父子组件中互相传递消息一样。

假设你有两个Vue页面需要通信: A 和 B ,A页面 在按钮上面绑定了点击事件,发送一则消息,想通知 B页面。

<!-- A.vue -->
<template>
    <button @click="sendMsg()">-</button>
</template>

<script> 
import { EventBus } from "../event-bus.js";
export default {
  methods: {
    sendMsg() {
      // 方法1
      EventBus.$emit("aMsg", '来自A页面的消息');

      // 方法2
      this.$EventBus.$emit("aMsg", '来自A页面的消息');
    }
  }
}; 
</script>

接下来,我们需要在 B页面 中接收这则消息。

<!-- B.vue -->
<template>
  <p>{{msg}}</p>
</template>

<script> 
import { 
  EventBus 
} from "../event-bus.js";
export default {
  data(){
    return {
      msg: ''
    }
  },
  mounted() {
     // 方法1
    EventBus.$on("aMsg", (msg) => {
      // A发送来的消息
      this.msg = msg;
    });

   // 方法2
   this.$EventBus.$on("aMsg", (msg) => {
      this.msg = msg;
    });
  }
};
</script>

同理我们也可以在 B页面 向 A页面 发送消息。这里主要用到的两个方法:

// 发送消息
EventBus.$emit(channel: string, callback(payload1,…))
EventBus.$emit(eventName, params)

// 监听接收消息
EventBus.$on(channel: string, callback(payload1,…))
EventBus.$on(eventName, callback(params))

如果使用不善,EventBus会是一种灾难,到底是什么样的“灾难”了?大家都知道vue是单页应用,如果你在某一个页面刷新了之后,与之相关的EventBus会被移除,这样就导致业务走不下去。还要就是如果业务有反复操作的页面,EventBus在监听的时候就会触发很多次,也是一个非常大的隐患。这时候我们就需要好好处理EventBus在项目中的关系。通常会用到,在vue页面销毁时,同时移除EventBus事件监听。
如果想移除事件的监听,可以像下面这样操作:

import { 
  eventBus 
} from './event-bus.js'


// ...
// 在b页面里取消监听接收,否则会数据重复  a最好也写下
beforeDestroyed(){
  EventBus.$off('aMsg', {})
  // this.$EventBus.$off('aMsg', {})
  // EventBus.$off()
}

$on事件是不会自动清楚销毁的,需要我们手动来销毁,否则在b组件每次加载一次就会创建一个监听,会重复监听到数据。
可以使用 EventBus.$off('aMsg')来移除应用内所有对此某个事件的监听。
或者直接调用 EventBus.$off()来移除所有事件频道,不需要添加任何参数 。

情况1:bus进行页面传参

总结: 所以,如果想要用bus 来进行页面组件之间的数据传递,需要注意两点:
1、组件A $emit事件应在beforeDestory生命周期内。
2、其次,组件B内的$on记得要销毁。



因为页面跳转的时候 ,a页面在之前已经emit了,但是b页面首次并没有created,b页面还监听不到。
可以把A页面组件中的emit事件写在beforeDestory中去。因为这个时候,B页面组件已经被created了,也就是我们写的$on事件已经可以触发了

所以可以,在beforeDestory的时候,$emit事件。例如:

  editList (index, date, item) {
//  点击进入编辑的页面,需要传递的参数比较多。      
      this.item = item.type
      this.date = date
      this.$router.replace({path: '/moneyRecord'})
    }

// 重新在data属性内部定义新的变量,来存储要传过去的数据;
然后:
 beforeDestroy () {

 bus.$emit('pushDataToBpage', {
        item: this.item,
        date: this.date
      })
 },
情况2:bus进行组件传参

如果是同一个页面,两个组件传参,A里啥时候用啥时候 emit 就行 不用放到beforeDestroy里
然后B组件里 在mounted 里 先$off$on
或者在B组件的 mounted里 $on 在beforeDestroy 里 $off
总之记得要手动销毁 bus就行

【参考eventbus实战记录://www.greatytc.com/p/fde85549e3b0

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,539评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,911评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,337评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,723评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,795评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,762评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,742评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,508评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,954评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,247评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,404评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,104评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,736评论 3 324
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,352评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,557评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,371评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,292评论 2 352