欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Vue中使用的EventBus有生命周期

 更新時間:2018年07月12日 14:18:44   投稿:mrr  
這篇文章主要介紹了Vue中使用的EventBus有生命周期的相關(guān)知識,非常不錯,具有一定的參考借鑒價值,需要的朋友可以參考下

最近遇到了vue項目中的性能問題,整個項目不斷的進行操作五分鐘左右,頁面已經(jīng)很卡,查看頁面占用了1.5G內(nèi)存,經(jīng)過排查一部分原因,是自己模塊使用的eventBus在離開頁面未進行off掉。我們進行下驗證:

1、不隨生命周期銷毀

我們在home首頁的代碼是這樣的:

 created () {
 let text = Array(1000000).fill('xxx').join(',') //創(chuàng)建一個大的字符串用于驗證內(nèi)存占用
 $eventBus.$on('home-on', (...args) => {
  this.text = text
 })
 },
 mounted () {
 setTimeout(() => {
  $eventBus.$emit('home-on', '這是home $emit參數(shù)', 'ee')
 }, 1000)
 },
 beforeDestroy () {
 // 注意這里沒有off掉'home-on'的訂閱事件
 }
 // eventBus是全局的

(1)在home頁時:我們拍個內(nèi)存快照查看下home頁的內(nèi)存占用:

圖1

一共17.4M我們創(chuàng)建出的字符串text占用了8M,這在home頁沒銷毀時是正常的

(2)離開home頁時:然后我們點擊跳轉(zhuǎn)到其他頁面離開home頁,然后再拍個內(nèi)存快照:

圖2

創(chuàng)建的'xxx,xxx...'字符串是home頁面掛載在this.text上的,離開了home依然存在著,查看下面的箭頭看到是存在了EventBus上,原因很明顯,是我們在beforeDestroy的時候沒把訂閱的事件給銷毀掉,而EventBus是全局的,造訂閱的on的回調(diào)里調(diào)用了this.text,因此回調(diào)里的this就一直掛在了EventBus里。

2、隨著生命周期銷毀

 created () {
 let text = Array(1000000).fill('xxx').join(',') //創(chuàng)建一個大的字符串用于驗證內(nèi)存占用
 $eventBus.$on('home-on', (...args) => {
  this.text = text
 })
 },
 mounted () {
 setTimeout(() => {
  $eventBus.$emit('home-on', '這是home $emit參數(shù)', 'ee')
 }, 1000)
 },
 beforeDestroy () {
 $eventBus.$off('home-on') // 注意這里off掉了'home-on'的訂閱事件
 }
 // eventBus是全局的

(1)在home頁時:內(nèi)存快照不用多說自然是圖1的內(nèi)存快照

(2)離開home頁時:再來拍個內(nèi)存快照:

發(fā)現(xiàn)減到了10M,且通過內(nèi)存占用看到'string'里已經(jīng)沒有'xxx,xxx...'的內(nèi)存占用了,這說明我們銷毀之后瀏覽器回收了this.text。

好,以上說這么多只是說明在使用EventBus時要時刻注意訂閱事件的銷毀。如果有一個還好,如果有5個,6個...也要挨個銷毀這就比較麻煩了。話說off銷毀這件重復性勞動這些都應該是機器做的事情,我們不應該去關(guān)心的。

基于以上我們自然就想到讓EventBus隨著生命周期銷毀就行了嘛。EventBus有生命周期的特性那么就離不開在使用.$on的this的關(guān)聯(lián),每個Vue組件有自己的_uid作為唯一標識,因此我們基于uid稍微改造下EventBus,讓EventBus和_uid關(guān)聯(lián)起來:

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所屬的數(shù)組里
 }
 $on (eventName, callback, vm) { // vm是在組件內(nèi)部使用時組件當前的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) // 注意第三個參數(shù)需要傳當前組件的this,如果不傳則需要手動銷毀
 },
 mounted () {
 setTimeout(() => {
  this.$eventBus.$emit('home-on', '這是home $emit參數(shù)', 'ee')
 }, 1000)
 },
 beforeDestroy () {
 // 這里就不需要手動的off銷毀eventBus訂閱的事件了
 }

總結(jié)

以上所述是小編給大家介紹的Vue中使用的EventBus有生命周期,希望對大家有所幫助,如果大家有任何疑問請給我留言,小編會及時回復大家的。在此也非常感謝大家對腳本之家網(wǎng)站的支持!

相關(guān)文章

  • 詳解vue-cli腳手架build目錄中的dev-server.js配置文件

    詳解vue-cli腳手架build目錄中的dev-server.js配置文件

    這篇文章主要介紹了詳解vue-cli腳手架build目錄中的dev-server.js配置文件,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2017-11-11
  • Vue.js結(jié)合bootstrap前端實現(xiàn)分頁和排序效果

    Vue.js結(jié)合bootstrap前端實現(xiàn)分頁和排序效果

    這篇文章主要為大家詳細介紹了Vue.js結(jié)合bootstrap 前端實現(xiàn)分頁和排序效果,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2018-12-12
  • vue-cli的webpack模板項目配置文件分析

    vue-cli的webpack模板項目配置文件分析

    本篇文章主要對vue-cli的webpack模板項目配置文件進行分析。具有很好的參考價值。下面跟著小編一起來看下吧
    2017-04-04
  • vue配置啟動項目自動打開瀏覽器方式

    vue配置啟動項目自動打開瀏覽器方式

    這篇文章主要介紹了vue配置啟動項目自動打開瀏覽器方式,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2022-04-04
  • 基于Vue3+ts封裝一個簡單版的Message組件

    基于Vue3+ts封裝一個簡單版的Message組件

    近日項目中需要使用信息提示框的功能,ui組件庫使用的是字節(jié)的arco-design-vue,看了一下,現(xiàn)有的Message不滿足要是需求,直接使用message組件的話,改樣式太麻煩,所以本文就本就介紹了基于Vue3+ts封裝一個簡單版的Message組件,需要的朋友可以參考下
    2023-09-09
  • 解決vue的router組件component在import時不能使用變量問題

    解決vue的router組件component在import時不能使用變量問題

    這篇文章主要介紹了解決vue的router組件component在import時不能使用變量問題,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2020-07-07
  • Vue中router-view無法顯示的解決辦法

    Vue中router-view無法顯示的解決辦法

    這篇文章主要給大家介紹了關(guān)于Vue中router-view無法顯示的解決辦法,router-view組件作為vue最核心的路由管理組件,在項目中作為路由管理經(jīng)常被使用到,需要的朋友可以參考下
    2023-07-07
  • vue腳手架創(chuàng)建項目時報catch錯誤及解決

    vue腳手架創(chuàng)建項目時報catch錯誤及解決

    這篇文章主要介紹了vue腳手架創(chuàng)建項目時報catch錯誤及解決方案,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2023-01-01
  • vue實現(xiàn)Input輸入框模糊查詢方法

    vue實現(xiàn)Input輸入框模糊查詢方法

    這篇文章主要為大家詳細介紹了vue實現(xiàn)Input輸入框模糊查詢方法,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2019-10-10
  • vue實現(xiàn)tab切換外加樣式切換方法

    vue實現(xiàn)tab切換外加樣式切換方法

    下面小編就為大家分享一篇vue實現(xiàn)tab切換外加樣式切換方法,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2018-03-03

最新評論