欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  IT编程

Vue中使用的EventBus有生命周期

程序员文章站 2022-03-21 20:35:32
最近遇到了vue项目中的性能问题,整个项目不断的进行操作五分钟左右,页面已经很卡,查看页面占用了1.5g内存,经过排查一部分原因,是自己模块使用的eventbus在离开页面...

最近遇到了vue项目中的性能问题,整个项目不断的进行操作五分钟左右,页面已经很卡,查看页面占用了1.5g内存,经过排查一部分原因,是自己模块使用的eventbus在离开页面未进行off掉。我们进行下验证:

1、不随生命周期销毁

我们在home首页的代码是这样的:

 created () {
 let text = array(1000000).fill('xxx').join(',') //创建一个大的字符串用于验证内存占用
 $eventbus.$on('home-on', (...args) => {
  this.text = text
 })
 },
 mounted () {
 settimeout(() => {
  $eventbus.$emit('home-on', '这是home $emit参数', 'ee')
 }, 1000)
 },
 beforedestroy () {
 // 注意这里没有off掉'home-on'的订阅事件
 }
 // eventbus是全局的

(1)在home页时:我们拍个内存快照查看下home页的内存占用:

Vue中使用的EventBus有生命周期

图1

一共17.4m我们创建出的字符串text占用了8m,这在home页没销毁时是正常的

(2)离开home页时:然后我们点击跳转到其他页面离开home页,然后再拍个内存快照:

Vue中使用的EventBus有生命周期

图2

创建的'xxx,xxx...'字符串是home页面挂载在this.text上的,离开了home依然存在着,查看下面的箭头看到是存在了eventbus上,原因很明显,是我们在beforedestroy的时候没把订阅的事件给销毁掉,而eventbus是全局的,造订阅的on的回调里调用了this.text,因此回调里的this就一直挂在了eventbus里。

2、随着生命周期销毁

 created () {
 let text = array(1000000).fill('xxx').join(',') //创建一个大的字符串用于验证内存占用
 $eventbus.$on('home-on', (...args) => {
  this.text = text
 })
 },
 mounted () {
 settimeout(() => {
  $eventbus.$emit('home-on', '这是home $emit参数', 'ee')
 }, 1000)
 },
 beforedestroy () {
 $eventbus.$off('home-on') // 注意这里off掉了'home-on'的订阅事件
 }
 // eventbus是全局的

(1)在home页时:内存快照不用多说自然是图1的内存快照

(2)离开home页时:再来拍个内存快照:

Vue中使用的EventBus有生命周期

发现减到了10m,且通过内存占用看到'string'里已经没有'xxx,xxx...'的内存占用了,这说明我们销毁之后浏览器回收了this.text。

好,以上说这么多只是说明在使用eventbus时要时刻注意订阅事件的销毁。如果有一个还好,如果有5个,6个...也要挨个销毁这就比较麻烦了。话说off销毁这件重复性劳动这些都应该是机器做的事情,我们不应该去关心的。

基于以上我们自然就想到让eventbus随着生命周期销毁就行了嘛。eventbus有生命周期的特性那么就离不开在使用.$on的this的关联,每个vue组件有自己的_uid作为唯一标识,因此我们基于uid稍微改造下eventbus,让eventbus和_uid关联起来:

class eventbus {
 constructor (vue) {
 if (!this.handles) {
  object.defineproperty(this, 'handles', {
  value: {},
  enumerable: false
  })
 }
 this.vue = vue
 this.eventmapuid = {} // _uid和eventname的映射
 }
 seteventmapuid (uid, eventname) {
 if (!this.eventmapuid[uid]) this.eventmapuid[uid] = []
 this.eventmapuid[uid].push(eventname) // 把每个_uid订阅的事件名字push到各自uid所属的数组里
 }
 $on (eventname, callback, vm) { // vm是在组件内部使用时组件当前的this用于取_uid
 if (!this.handles[eventname]) this.handles[eventname] = []
 this.handles[eventname].push(callback)
 if (vm instanceof this.vue) this.seteventmapuid(vm._uid, eventname)
 }
 $emit () {
 // console.log('eventbus emit eventname===', eventname)
 let args = [...arguments]
 let eventname = args[0]
 let params = args.slice(1)
 if (this.handles[eventname]) {
  let len = this.handles[eventname].length
  for (let i = 0; i < len; i++) {
  this.handles[eventname][i](...params)
  }
 }
 }
 $offvmevent (uid) {
 let currentevents = this.eventmapuid[uid] || []
 currentevents.foreach(event => {
  this.$off(event)
 })
 }
 $off (eventname) {
 delete this.handles[eventname]
 }
}
// 下面写成vue插件形式,直接引入然后vue.use($eventbus)进行使用
let $eventbus = {}
$eventbus.install = (vue, option) => {
 vue.prototype.$eventbus = new eventbus(vue)
 vue.mixin({
 beforedestroy () {
  this.$eventbus.$offvmevent(this._uid) // 拦截beforedestroy钩子自动销毁自身所有订阅的事件
 }
 })
}
export default $eventbus

下面来进行使用

// main.js中
...
import eventbus from './eventbus.js'
vue.use(enemtbus)
...

组件中使用:

 created () {
 let text = array(1000000).fill('xxx').join(',')
 this.$eventbus.$on('home-on', (...args) => {
  console.log('home $on====>>>', ...args)
  this.text = text
 }, this) // 注意第三个参数需要传当前组件的this,如果不传则需要手动销毁
 },
 mounted () {
 settimeout(() => {
  this.$eventbus.$emit('home-on', '这是home $emit参数', 'ee')
 }, 1000)
 },
 beforedestroy () {
 // 这里就不需要手动的off销毁eventbus订阅的事件了
 }

总结

以上所述是小编给大家介绍的vue中使用的eventbus有生命周期,希望对大家有所帮助