组件通信包括:父子组件间的通讯和兄弟组件间的通讯。在组件化系统构建中,组件间通讯必不可少的 (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记得要销毁。