Vue3中自定義事件總線的實(shí)現(xiàn)代碼
前言
在 Vue 開(kāi)發(fā)中,組件之間的通信是一個(gè)常見(jiàn)的需求,無(wú)論是父組件向子組件傳遞數(shù)據(jù),還是子組件向父組件傳遞數(shù)據(jù),甚至是兄弟組件之間的數(shù)據(jù)交換。這些通信需求在構(gòu)建復(fù)雜的 Vue 應(yīng)用時(shí)尤為關(guān)鍵。
Vue 提供了多種組件通信的方式,如 props 用于父組件向子組件傳遞數(shù)據(jù),emit 用于子組件觸發(fā)事件并傳遞數(shù)據(jù)給父組件,vuex 適用于狀態(tài)管理場(chǎng)景,而 provide/inject 則提供了依賴注入的方式。在 Vue 2 中,還有 eventBus 和 $attrs/$listeners 以及 $parent/children 等方法來(lái)輔助組件間的通信。
然而,隨著 Vue 3 的發(fā)布,一些在 Vue 2 中常用的通信方式在 Vue 3 中可能不再適用或有所變化。為了應(yīng)對(duì)這種變化,我們可以選擇在 Vue 3 應(yīng)用中實(shí)現(xiàn)自定義的“事件總線”機(jī)制。這種機(jī)制通常是將一個(gè)事件中心(或稱為事件總線)掛載到Vue的全局對(duì)象上,從而使得任何組件都可以方便地通過(guò)事件總線來(lái)發(fā)布或監(jiān)聽(tīng)事件。
通過(guò)使用這樣的自定義事件總線,開(kāi)發(fā)者可以在 Vue 3 應(yīng)用中實(shí)現(xiàn)靈活的組件間通信,無(wú)論這些組件之間的層級(jí)關(guān)系如何,都能輕松地實(shí)現(xiàn)數(shù)據(jù)和事件的傳遞。
發(fā)布-訂閱模式
發(fā)布-訂閱模式(Publish-Subscribe Pattern)是一種在軟件設(shè)計(jì)中常見(jiàn)的模式,它允許消息發(fā)送者和接收者之間通過(guò)事件進(jìn)行通信,而不必直接相互依賴。這種模式的主要思想是通過(guò)一個(gè)被稱為“消息中心”或“事件總線”的實(shí)體來(lái)協(xié)調(diào)消息的發(fā)布和訂閱。
一個(gè)完整的發(fā)布-訂閱模式通常包含以下幾個(gè)部分:
- 發(fā)布者(Publisher):負(fù)責(zé)向消息中心發(fā)布事件或消息的對(duì)象。發(fā)布者通常不關(guān)心誰(shuí)訂閱了這些事件,只負(fù)責(zé)在特定情況下觸發(fā)它們。
- 訂閱者(Subscriber):對(duì)特定事件感興趣的對(duì)象,它們會(huì)向消息中心訂閱這些事件。當(dāng)發(fā)布者發(fā)布一個(gè)事件時(shí),所有訂閱了該事件的訂閱者都會(huì)收到通知。
- 消息中心(Event Bus/Message Center):負(fù)責(zé)管理事件的發(fā)布、訂閱和通知的對(duì)象。它存儲(chǔ)了事件和訂閱者之間的關(guān)系,并在事件被發(fā)布時(shí),將事件通知給所有訂閱了該事件的訂閱者。

發(fā)布-訂閱模式的優(yōu)點(diǎn)包括:
- 解耦性:發(fā)布者和訂閱者之間不存在直接的依賴關(guān)系,這使得它們可以獨(dú)立地變化,而不需要修改對(duì)方。這種解耦性提高了代碼的可維護(hù)性和可擴(kuò)展性。
- 可擴(kuò)展性:可以輕松地添加新的發(fā)布者和訂閱者,而無(wú)需修改現(xiàn)有的代碼。這使得系統(tǒng)能夠靈活地適應(yīng)不斷變化的需求。
- 靈活性:支持多個(gè)訂閱者同時(shí)訂閱同一個(gè)事件,并且可以根據(jù)需要定制事件的處理方式。這種靈活性使得系統(tǒng)能夠應(yīng)對(duì)各種復(fù)雜場(chǎng)景。
- 時(shí)間解耦:發(fā)布者可以在任何時(shí)刻發(fā)布事件,而訂閱者則可以在自己方便的時(shí)候處理這些事件。這種時(shí)間解耦性使得系統(tǒng)更加靈活和高效。
綜上所述,發(fā)布-訂閱模式提供了一種強(qiáng)大而靈活的方式來(lái)處理組件之間的通信和協(xié)作,使得系統(tǒng)更加健壯、可維護(hù)和可擴(kuò)展。在Vue.js 等現(xiàn)代前端框架中,發(fā)布-訂閱模式被廣泛應(yīng)用于組件之間的通信和狀態(tài)管理。
實(shí)現(xiàn)發(fā)布-訂閱模式
在深入探討發(fā)布-訂閱模式時(shí),其核心機(jī)制在于一個(gè)精心構(gòu)建的事件中心。這個(gè)事件中心不僅作為消息的中轉(zhuǎn)站,還承載著存儲(chǔ)事件和訂閱者之間關(guān)系的重要職責(zé)。
事件中心
定義一個(gè) EventBus 類(lèi),用于存儲(chǔ)事件和訂閱者關(guān)系,代碼如下:
interface EventType {
readonly callback: Function
readonly once: boolean
}
type EventsType = Record<string, EventType[]>
class EventEmitter {
private events: EventsType = {};
}
events 是一個(gè)對(duì)象,其中每個(gè)鍵都是事件名稱,值是一個(gè)由 EventType 對(duì)象組成的數(shù)組,EventType 對(duì)象中包含是否只訂閱一次標(biāo)志位和回調(diào)函數(shù)。
發(fā)布事件
定義一個(gè) emit 方法,用于發(fā)布事件,代碼如下:
type EventArgs = Record<string, number | string | object | boolean>
interface EventType {
readonly callback: Function
readonly once: boolean
}
type EventsType = Record<string, EventType[]>
class EventEmitter {
private events: EventsType = {}
emit(eventName: string, eventArgs: EventArgs) {
eventName?.split(',').forEach((eventKey: string) => {
eventKey = eventKey.trim()
const events = this.events[eventKey] || []
let { length } = events
for (let i = 0; i < length; i++) {
const { callback, once } = events[i]
if (once) {
events.splice(i, 1)
if (length === 0) {
delete this.events[eventKey]
}
length--
i--
}
callback.apply(this, [eventArgs])
}
})
}
}
emit 方法接收兩個(gè)參數(shù),第一個(gè)參數(shù) eventName 為事件名稱,第二個(gè)參數(shù) eventArgs 為事件參數(shù)。其中 eventName 參數(shù)可以是一個(gè)以逗號(hào)分隔的字符串,表示同時(shí)發(fā)布多個(gè)事件。核心邏輯便是遍歷 events 對(duì)象,找到對(duì)應(yīng)的事件名稱,然后遍歷事件名稱對(duì)應(yīng)的事件數(shù)組,依次通過(guò)調(diào)用 apply 方法,執(zhí)行回調(diào)函數(shù)。
訂閱事件
定義一個(gè) on 方法,用于訂閱事件,代碼如下:
class EventEmitter {
private events: EventsType = {}
on(eventName: string, callback: CallbackType, once?: boolean) {
eventName?.split(',').forEach((eventKey: string) => {
eventKey = eventKey.trim()
if (!this.events[eventKey]) {
this.events[eventKey] = [];
} else {
this.events[eventKey].push({
callback,
once: !!once
})
}
})
return this
}
}
on 方法接收三個(gè)參數(shù),第一個(gè)參數(shù) eventName 為事件名稱,第二個(gè)參數(shù) callback 為回調(diào)函數(shù),第三個(gè)參數(shù) once 表示是否只訂閱一次。核心邏輯是遍歷 eventName 參數(shù) split 之后的數(shù)組對(duì)象(允許同時(shí)監(jiān)聽(tīng)多個(gè)事件,多個(gè)事件之間以逗號(hào)分隔),將事件名稱拆分成數(shù)組,然后遍歷數(shù)組,將回調(diào)函數(shù)存入 events 對(duì)象中。
取消訂閱事件
定義一個(gè) off 方法,用于取消訂閱事件,代碼如下:
class EventEmitter {
private events: EventsType = {}
off(eventName: string, callback?: CallbackType) {
// 如果不傳callback,就清除所有事件
if (!eventName) {
this.events = {}
}
eventName?.split(',').forEach((eventKey: string) => {
if(!callback) {
delete this.events[eventKey]
}
const events = this.events[eventKey] || []
let { length } = events
for (let i = 0; i < length; i++) {
if (events[i].callback === callback) {
events.splice(i, 1)
length --
i --
}
}
if (events.length === 0) {
delete this.events[eventKey]
}
})
return this
}
}
off 方法接收兩個(gè)參數(shù),第一個(gè)參數(shù) eventName 為事件名稱,第二個(gè)參數(shù) callback 為回調(diào)函數(shù),核心邏輯是遍歷 eventName 參數(shù) split 之后的數(shù)組對(duì)象(允許同時(shí)取消訂閱多個(gè)事件,多個(gè)事件之間以逗號(hào)分隔),將事件名稱拆分成數(shù)組,然后遍歷數(shù)組,將回調(diào)函數(shù)從數(shù)組中刪除。
只訂閱一次
定義一個(gè) once 方法,用于只訂閱一次,代碼如下:
class EventEmitter {
private events: EventsType = {}
once(eventName: string, callback: CallbackType) {
return this.on(eventName, callback, true)
}
}
once 方法接收兩個(gè)參數(shù),第一個(gè)參數(shù) eventName 為事件名稱,第二個(gè)參數(shù) callback 為回調(diào)函數(shù)。once 方法內(nèi)部調(diào)用 on 方法,并將第三個(gè)參數(shù)設(shè)置為 true,表示只訂閱一次。
如何在 Vue 中使用
當(dāng)我們想要在 Vue 應(yīng)用中使用發(fā)布-訂閱模式時(shí),通常會(huì)引入一個(gè)全局的事件總線 (Event Bus) 來(lái)作為通信的中心。這樣,無(wú)論組件之間有著怎樣的層級(jí)關(guān)系,它們都可以輕松地通過(guò)事件總線進(jìn)行通信。
為了在 Vue 應(yīng)用中實(shí)現(xiàn)這一功能,我們需要在應(yīng)用的入口文件(通常是 main.ts 或 main.js,取決于你的項(xiàng)目配置和所使用的 TypeScript 或 JavaScript)中引入并實(shí)例化事件總線。然后,我們可以利用 Vue 的 provide 方法將事件總線注冊(cè)為全局對(duì)象,使得在 Vue 應(yīng)用的任何組件中都能通過(guò) inject 來(lái)訪問(wèn)它。
在 main.ts 中編寫(xiě)代碼如下:
import { createApp } from 'vue'
import { EventEmitter } from '@qftjs/tiny-editor-core'
import App from './App.vue'
const app = createApp(App)
const bus = new EventEmitter()
app.provide('$bus', bus)
app.mount('#app')
在 Vue 組件中通過(guò) inject 方法注入 $bus 對(duì)象,然后就可以使用 $bus 對(duì)象進(jìn)行事件訂閱和事件發(fā)布。具體代碼如下:
<script setup lang="ts">
import { inject } from 'vue'
const $bus = inject('$bus')
$bus.emit('dragStart')
</script>
以上就是Vue3中自定義事件總線的實(shí)現(xiàn)代碼的詳細(xì)內(nèi)容,更多關(guān)于Vue3自定義事件總線的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
Vue v-model相關(guān)知識(shí)總結(jié)
這篇文章主要介紹了Vue ​v-model相關(guān)知識(shí)總結(jié),幫助大家更好的理解和學(xué)習(xí)使用vue框架,感興趣的朋友可以了解下2021-01-01
淺析Vue中defineProps的解構(gòu)和不解構(gòu)
defineProps?是?Vue?3?Composition?API?中用來(lái)聲明組件接收的?props?的方法,本文主要為大家介紹了defineProps的解構(gòu)和不解構(gòu),感興趣的可以了解下2025-02-02
十個(gè)有用的自定義Vue鉤子函數(shù)總結(jié)
這篇文章主要為大家介紹了十個(gè)Vue.js中有用的自定義鉤子,讓我們的代碼更加好看。文中的示例代碼講解詳細(xì),感興趣的小伙伴可以了解一下2022-04-04
vue自定義插件封裝,實(shí)現(xiàn)簡(jiǎn)易的elementUi的Message和MessageBox的示例
這篇文章主要介紹了vue自定義插件封裝,實(shí)現(xiàn)簡(jiǎn)易的elementUi的Message和MessageBox的示例,幫助大家更好的理解和使用vue框架,感興趣的朋友可以了解下2020-11-11
uniapp實(shí)現(xiàn)省市區(qū)三級(jí)級(jí)聯(lián)選擇功能(含地區(qū)json文件)
這篇文章主要給大家介紹了關(guān)于uniapp實(shí)現(xiàn)省市區(qū)三級(jí)級(jí)聯(lián)選擇功能(含地區(qū)json文件)的相關(guān)資料,級(jí)級(jí)聯(lián)是一種常見(jiàn)的網(wǎng)頁(yè)交互設(shè)計(jì),用于省市區(qū)選擇,它的目的是方便用戶在一系列選項(xiàng)中進(jìn)行選擇,并且確保所選選項(xiàng)的正確性和完整性,需要的朋友可以參考下2024-06-06
使用watch監(jiān)聽(tīng)對(duì)象里面值的變化
這篇文章主要介紹了使用watch監(jiān)聽(tīng)對(duì)象里面值的變化,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-01-01
Vue3在Setup中使用axios請(qǐng)求獲取的值方式
這篇文章主要介紹了Vue3在Setup中使用axios請(qǐng)求獲取的值方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-06-06
vue favicon設(shè)置以及動(dòng)態(tài)修改favicon的方法
這篇文章主要介紹了vue favicon設(shè)置以及動(dòng)態(tài)修改favicon的方法,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2018-12-12
vue實(shí)現(xiàn)打印指定組件內(nèi)容的示例詳解
這篇文章主要和大家分享一下vue中打印指定組件內(nèi)容,多頁(yè)打印自動(dòng)適配紙張大小打印的方案,文中的示例代碼講解詳細(xì),需要的可以參考一下2024-03-03

