vue中eventbus被多次触发(vue中使用eventbus踩过的坑)

2020-01-08 13:30:30

在做vue evenbus的时候,我们不面会遇到多测触发监听的问题,不但重新生成或者多次生成组件,以及开发热加载模式中都会发生

不过这些问题,跨组件调用方法的神奇,supervue已经为我们解决了这些问题。我们可以去用一下。

点击查看supervue用法

下面我们来解释下为什么会出现上面所说的问题。


问题参考地址 vue中eventbus被多次触发(vue中使用eventbus踩过的坑)


一开始的需求是这样子的,我为了实现两个页面组件之间的数据传递,假设我有页面A,点击页面A上的某一个按钮之后,页面会自动跳转到页面B,同时我希望将页面A上的某一些参数携带过去给页面B。


然后我就想,这不就是不同组件之间的数据传递问题而已吗?直接用bus 巴士事件来传递数据不就行了吗。于是,我就很愉快地进行了。关于vue中的eventbus的使用,我之前在一篇vue中的数据传递中有提到过。 



先给你们看一下我一开始的代码:


实现目标: 

点击之后,bus emit事件,然后顺便跳转路由到/moneyRecord页面。

接下来就是在MoneyRecord页面中去on接收这个事件,然后接受参数。

// 这是页面A的内部触发bus事件的代码

  editList (index, date, item) {

//  点击进入编辑的页面,需要传递的参数比较多。

      console.log(index, date, item)

      bus.$emit('get', {

        item: item.type,

        date: date

      })

      this.$router.replace({path: '/moneyRecord'})

    }

 

// moneyRecord页面

created () {

    //这里我将icon的list给保存下来了

    bus.$on('get', this.myhandle)

  },

methods: {

  myhandle (val) {

      console.log(val, '这是从上个页面传递过来的参数')

    }

}

就当我欣喜若狂的时候,觉得自己只要在页面A触发了get事件,页面B中就会理所当然的接受了数据。然而,结果却不如人意,看一下下面的动图。


主要是看“”这是从上个页面传来的数据这一行数据的输出次数情况来判断事件触发次数。“”



test.gif

不知道你有没有发现,就是我第一次进去list页面的时候,我随便点击一下list下的任何一个item,控制台没有输出。但是当我第二次再点击触发事件的时候,就会输出一个测试数据。再一次进去点击,就输出两个数据。。。依次增加了。(控制台上那个“这是从上个页面传来的数据”就是测试数据)


所以,有两个问题。


问题:


问题1: 为什么第一次触发的时候页面B中的on事件没有被触发

问题2: 为什么后面再一次依次去触发的时候会出现,每一次都会发现好像之前的on事件分发都没有被撤销一样,导致每一次的事件触发执行越来越多。

解决


针对问题1

这个还得从vue的生命周期说起了,我先进行了测试,就是当从页面组件A跳转到页面组件B的时候,两个组件的生命周期分别是怎么样的,关于vue的生命周期具体每一个时期做什么事情我就不再赘述了,下面po一张vue生命周期的图。



image.png

我自己做了实验来验证,这个页面跳转过程中,这两个组件的生命周期的执行情况。


// 我分别在页面A和页面B中去添加以下代码:

beforeCreate () {

   console.group('%c%s', 'color:red', 'beforeCreate 创建前状态===============组件2》')

 },

 created () {

   console.group('%c%s', 'color:red', 'created 创建完毕状态===============组件2》')

 },

 beforeMount () {

   console.group('%c%s', 'color:red', 'beforeMount 挂载前状态===============组件2》')

 },

 mounted () {

   console.group('%c%s', 'color:red', 'mounted 挂载状态===============组件2》')

 },

 beforeUpdate () {

   console.group('%c%s', 'color:red', 'beforeUpdate 更新前状态===============组件2》')

 },

 updated () {

   console.group('%c%s', 'color:red', 'updated 更新状态===============组件2》')

 },

 beforeDestroy () {

   console.group('%c%s', 'color:red', 'beforeDestroy 破前状态===============组件2》')

 },

 destroyed () {

   console.group('%c%s', 'color:red', 'destroyed 破坏状态===============组件2》')

 }

// 另外一个组件的我就不放出来了

测试结果图:



test.gif


image.png

其实,可以通过结果清楚看到,当我们还在页面A的时候,页面B还没生成,也就是页面B中的 created中所监听的来自于A中的事件还没有被触发。这个时候当你A中emit事件的时候,B其实是没有监听到的。


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


所以,我们可以把A页面组件中的emit事件写在beforeDestory中去。因为这个时候,B页面组件已经被created了,也就是我们写的$on事件已经触发了


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


// 修改一下A页面中的代码:

// 这是原先的代码

  editList (index, date, item) {

//  点击进入编辑的页面,需要传递的参数比较多。

      console.log(index, date, item)

      this.item = item.type

      this.date = date

      this.$router.replace({path: '/moneyRecord'})

    }

// 重新在data属性内部定义新的变量,来存储要传过去的数据;

然后:

 beforeDestroy () {

 console.log(this.highlight, '这是今年的数据', this, '看看组件销毁之前会发生什么')

 bus.$emit('get', {

        item: this.item,

        date: this.date

      })

 },

接下来。看一下修改之后的效果



test.gif

可以看到,就是第一次点击list的时候,也就是第一次触发emit事件的时候,控制太就输出了,所以在beforeDestoryed去$emit是起到作用的,B页面组件也监听$on到了。


但是,好像,就是事件的触发还是会依次增加,就是控制台的输出每次都有所增加了。。。


解决:

看一下github上提出的。issue

https://github.com/vuejs/vue/issues/3399



image.png

尤大大提出了以下解决:



image.png

*就是说,这个$on事件是不会自动清楚销毁的,需要我们手动来销毁。(不过我不太清楚这里的external bus 是什么意思,有大神能解答一下的吗,尤大大也提到如果是注册的是external bus 的时候需要清除)****


所以。我在B组件页面中添加Bus.$off来关闭。代码如下:


// 在B组件页面中添加以下语句,在组件beforeDestory的时候销毁。

  beforeDestroy () {

    bus.$off('get', this.myhandle)

  },

来看一下输出的结果


  • 2021-02-03 16:57:34

    iOS中的动态库和静态库分析

    由于最近研究组件化后调试时二进制映射源码的功能,发现需要对开发中的动态库和静态库需要有一些了解。所以就有了这篇文章,由于只是了解,并没有深入到编译层面,所以本篇文章只是简单了解一些库的知识,并不深入。

  • 2021-02-03 16:58:39

    iOS静态库与动态库的区别与打包

    这篇主要是记录一下 iOS 下静态库与动态库的打包流程,以便以后用到时快速查阅,供自己也供大家学习记录。同时也简述了一下 动态库 与 静态库 的区别。

  • 2021-02-03 16:59:59

    iOS 静态库和动态库全分析

    库就是程序代码的集合,将 N 个文件组织起来,是共享程序代码的一种方式。从本质上来说是一种可执行代码的二进制格式,可以被载入内存中执行。

  • 2021-02-03 17:01:30

    iOS库 .a与.framework区别

    静态库:连接时完整地拷贝至可执行文件中,被屡次使用就有多份冗余拷贝。 动态库:连接时不复制,程序运行时由系统动态加载到内存,供程序调用,系统只加载一次,多个程序共用,节省内存。

  • 2021-02-03 17:13:58

    iOS - 封装静态库

    静态库:链接时完整的拷贝至可执行文件中,被多次使用就有多份冗余拷贝,.a的静态库 .framework的静态库

  • 2021-02-03 17:16:07

    iOS 中的动态库、静态库和 framework

    首先来看什么是库,库(Library)说白了就是一段编译好的二进制代码,加上头文件就可以供别人使用。 什么时候我们会用到库呢?一种情况是某些代码需要给别人使用,但是我们不希望别人看到源码,就需要以库的形式进行封装,只暴露出头文件。另外一种情况是,对于某些不会进行大的改动的代码,我们想减少编译的时间,就可以把它打包成库,因为库是已经编译好的二进制了,编译的时候只需要 Link 一下,不会浪费编译时间。

  • 2021-02-03 17:17:53

    iOS 同一个工程下打包不同的app

    应用图标,启动画面,应用启动后的首页都不一样。 有些课程(例如公务员考试和高考)是有目标考试的概念,不同的目标考试大纲是不一样的。拿高考来举例,北京的高考和上海的高考,就有着完全不一样的考试大纲。高考的文科和理科,又有着完全不同的考试科目。 有些课程会有一些自定义的界面,例如高考的应用可以设置昵称,有些课程的真题练习中是有推荐真题模块的,而有些课程又没有。 有些课程有扫描答题卡功能,有些课程有考前冲刺功能,有些课程有大题专项查看功能,而有些课程又没有上述功能。另外还有一些微小细节,但是解决方法和类似,所以就不一一展开说明。

  • 2021-02-04 14:02:30

    window软件界面找不到了跑到屏幕外面去了

    一般可以这样操作,按Alt+空格,然后按M,然后用上下左右键把窗口移动到能看到的地方,再按回车。有些第三方的软件可能不能用,大部分都可以这样做。

  • 2021-02-04 14:08:13

    基于 Electron 的爬虫框架 Nightmare

    Electron 可以让你使用纯 JavaScript 调用 Chrome 丰富的原生的接口来创造桌面应用。你可以把它看作一个专注于桌面应用的 Node.js 的变体,而不是 Web 服务器。其基于浏览器的应用方式可以极方便的做各种响应式的交互,接下来介绍下关于 Electron 上衍生出的框架 Nightmare