vue组件通讯&&v兄弟组件通讯eventbus遇到的问题(屡次触发、第一次不触发)

组件通信包括:父子组件间的通讯和兄弟组件间的通讯。在组件化系统构建中,组件间通讯必不可少的 (vuex之后再说)。vue

父组件--> 子组件

1. 属性设置

父组件关键代码以下:vuex

1 <template>
2     <Child :child-msg="msg"></Child>
3 </template>

子组件关键代码以下:app

1 export default { 2   name: 'child', 3  props: { 4     child-msg: String 5  } 6 };

child-msg 为父组件给子组件设置的额外属性值,属性值需在子组件中设置props,子组件中可直接使用child-msg变量。函数

2. 子组件调用父组件

子组件经过 $parent 得到父组件,经过 $root 得到最上层的组件。组件化

子组件--> 父组件

1. 发送事件/监听事件

子组件中某函数内发送事件:this

this.$emit('toparentevent', 'data');

父组件监听事件:spa

<Child @toparentevent="toparentevent"></Child>
// 在methods里接收
methods: {  toparentevent(msg) { // msg就是传递的数据
 console.log(msg) } } 

toparentevent 为子组件自定义发送事件名称,父组件中@toparentevent为监听事件,toparentevent为父组件处理方法。code

2. 父组件直接获取子组件属性或方法

给要调用的子组件起个名字。将名字设置为子组件 ref 属性的值。component

<!-- 子组件。 ref的值是组件引用的名称 -->
<child-component ref="aName"></child-component>

父组件中经过 $refs.组件名 来得到子组件,也就能够调用子组件的属性和方法了。router

// 获取child.属性
this.$refs.aName.child
// 调用child.方法()
this.$refs.aName.child()

父组件经过 $children 能够得到全部直接子组件(父组件的子组件的子组件不是直接子组件)。须要注意 $children 并不保证顺序,也不是响应式的。

eventBus中央通讯

各组件可本身定义好组件内接收外部组件的消息事件便可,不用理会是哪一个组件发过来;而对于发送事件的组件,亦不用理会这个事件到底怎么发送给我须要发送的组件。

先设 main.js 置Bus

// main.js中
new Vue({ el: '#app', router, template: '<App/>', components: { App }, data: { eventBus: new Vue() } })

组件内监听事件:

1 created() { 2     this.$root.eventBus.$on('childa-message', function(data) { 3       console.log(data); 4  }); 5   }

发送事件的组件:

this.$root.eventBus.$emit('childa-message', this.data)
可能存在以下问题:
1.第一次触发的时候页面中的on事件没有被触发
2. 为何后面再一次依次去触发的时候会出现,每一次都会发现好像以前的on事件分发都没有被撤销同样,致使每一次的事件触发执行愈来愈多  

解答:

1. 当咱们还在页面A的时候,页面B还没生成,也就是页面B中的 created中所监听的来自于A中的事件尚未被触发。这个时候当你A中emit事件的时候,B实际上是没有监听到的,

    当你从页面A到页面B跳转的时候,发生了什么?首先是先B组件先created而后beforeMount接着A组件才被销毁,A组件才执行beforeDestory,以及destoryed

    因此,咱们能够把A页面组件中的emit事件写在beforeDestory中去。由于这个时候,B页面组件已经被created了,也就是咱们写的$on事件已经触发了

    修改以下:在beforeDestory生命周期里,$emit事件

1 beforeDestroy () { 2  this.$root.eventBus.$emit('child-message',this.data) 3  }

 2.  由于B页面$on事件是不会自动清楚销毁的,须要咱们手动来销毁,因此我在B组件页面中添加Bus.$off来关闭,销毁后就不存在执行屡次的状况了   。代码以下: 

1 // 在B组件页面中添加如下语句,在组件beforeDestory的时候销毁。 2  beforeDestroy () { 3     this.root.eventBus.$off('child-message')
4 },

               

总结: 因此,要用eventBus 来进行页面组件之间的数据传递,须要注意亮点,组件A$emit事件应在beforeDestory生命周期内。其次,组件B内的$on记得要销毁。

相关文章
相关标签/搜索