vue组件非常常见的有父子组件通信,兄弟组件通信。而父子组件通信就很简单,父组件会通过 props 向下传数据给子组件,当子组件有事情要告诉父组件时会通过 $emit 事件告诉父组件。今天就来说说如果两个页面没有任何引入和被引入关系,该如何通信?
如果咱们的应用程序不需要类似Vuex这样的库来处理组件之间的数据通信,就可以考虑Vue中的 事件总线 ,即 EventBus来通信。
EventBus 又称为事件总线。在Vue中可以使用 EventBus 来作为沟通桥梁的概念,就像是所有组件共用相同的事件中心,可以向该中心注册发送事件或接收事件,所以组件都可以上下平行地通知其他组件,但也就是太方便所以若使用不慎,就会造成难以维护的“灾难”,因此才需要更完善的Vuex作为状态管理中心,将通知的概念上升到共享状态层次。
前面提到过,如果使用不善,EventBus会是一种灾难,到底是什么样的“灾难”了?大家都知道vue是单页应用,如果你在某一个页面刷新了之后,与之相关的EventBus会被移除,这样就导致业务走不下去。还有就是如果业务有反复操作的页面,EventBus在监听的时候就会触发很多次,也是一个非常大的隐患。这时候我们就需要好好处理EventBus在项目中的关系。通常会用到,在vue页面销毁时,同时移除EventBus事件监听。
初始化
- 首先需要创建事件总线并将其导出,以便其它模块可以使用或者监听它。我们可以通过两种方式来处理。先来看第一种,新创建一个 .js 文件,比如 event-bus.js
// event-bus.js
import Vue from 'vue'
export const EventBus = new Vue()
实质上EventBus是一个不具备 DOM 的组件,它具有的仅仅只是它实例方法而已,因此它非常的轻便。
另外一种方式,可以直接在项目中的 main.js 初始化 EventBus :
// main.js
Vue.prototype.$EventBus = new Vue()
注意,这种方式初始化的EventBus是一个全局的事件总线。用法基本一样,全局的不用引入event-bus.js,只需
this.$EventBus
就可以调用$emit、$on、$off
等方法。现在我们已经创建了 EventBus ,接下来你需要做到的就是在你的组件中加载它,并且调用同一个方法,就如你在父子组件中互相传递消息一样。
// 发送消息
EventBus.$emit(channel: string, callback(payload1,…))
// 监听接收消息
EventBus.$on(channel: string, callback(payload1,…))
// 移除事件的监听
EventBus.$off('aMsg')
// 移除所有事件的监听
EventBus.$off()
实例:
// eventBus.js
import Vue from 'vue';
export const eventBus = new Vue();
// CptA.vue
<template>
<div>
<button @click="btn">btn</button>
</div>
</template>
<script>
import {eventBus} from "../utils/eventBus";
export default {
name: "CptA",
methods: {
btn() {
eventBus.$emit('aMsg', '我来自CptA')
}
},
}
</script>
// CptB.vue
<template>
<div>{{msg}}</div>
</template>
<script>
import {eventBus} from "../utils/eventBus";
export default {
name: "PageA",
data() {
return {
msg: 'value'
}
},
mounted() {
eventBus.$on('aMsg', mm => {
this.msg = mm;
console.log(mm)
})
}
}
</script>
组件之间使用this.$bus.$on传值之前需要先this.$bus.$off注销事件,否则会导致多次监听问题
this.$bus.$off('toOrder').$on('toOrder', this.updateAddress)