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


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

    父组件--> 子组件

    1. 属性设置

    父组件关键代码如下:

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

    子组件关键代码如下:

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

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

    2. 子组件调用父组件

    子组件通过 $parent 获得父组件,通过 $root 获得最上层的组件。

    子组件--> 父组件

    1. 发送事件/监听事件

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

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

    父组件监听事件:

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

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

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

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

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

    父组件中通过 $refs.组件名 来获得子组件,也就可以调用子组件的属性和方法了。

    // 获取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记得要销毁。

  • 相关阅读:
    优化慢执行或慢查询的方法
    Top K问题的两种解决思路
    优先队列实现 大小根堆 解决top k 问题
    进程间的八种通信方式----共享内存是最快的 IPC 方式
    二叉树基础之按层打印
    按层打印二叉树--每行打印一层
    给定一颗完全二叉树,给每一层添加上next的指针,从左边指向右边
    缓存与数据库一致性保证
    一致性哈希算法原理
    Linux复制指定目录下的文件夹结构
  • 原文地址:https://www.cnblogs.com/ljx20180807/p/9778782.html
Copyright © 2020-2023  润新知