news 2026/4/18 9:53:44

Vue 3 中使用 Mitt 事件总线

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
Vue 3 中使用 Mitt 事件总线

Vue3中使用Mitt事件总线实现组件通信。


Mitt是一个轻量级的事件库,可作为全局事件总线替代方案。


文章介绍了基本使用方法:创建事件总线实例、定义事件类型、在组件中发射和监听事件,并强调组件卸载时需取消监听以避免内存泄漏。


还展示了高级用法:工厂模式创建多总线、封装Vue插件、CompositionAPI封装Hook。


最后对比了Mitt与Vue原生通信方式的差异,建议谨慎使用事件总线,优先考虑props/emit或provide/inject等更明确的通信方式。


Vue 3 中使用 Mitt 事件总线

Mitt 是一个轻量级的事件发射/监听库,在 Vue 3 中可以作为全局事件总线的替代方案。

安装

npm install mitt # 或 yarn add mitt # 或 pnpm add mitt

基本使用

1.创建事件总线实例

方式一:创建全局事件总线

// src/utils/eventBus.ts import mitt from 'mitt' type Events = { // 定义事件类型 'user-login': { userId: string; username: string } 'user-logout': void 'notification': string 'update-count': number // 使用通配符监听所有事件 '*': { type: string; payload?: any } } const eventBus = mitt<Events>() export default eventBus

方式二:在 Composition API 中使用

vue

<!-- ComponentA.vue --> <script setup lang="ts"> import { onUnmounted } from 'vue' import eventBus from '@/utils/eventBus' // 发射事件 const emitLogin = () => { eventBus.emit('user-login', { userId: '123', username: '张三' }) } // 监听事件 eventBus.on('notification', (message) => { console.log('收到通知:', message) }) // 监听所有事件 eventBus.on('*', (type, payload) => { console.log(`事件类型: ${type}`, payload) }) // 组件卸载时取消监听 onUnmounted(() => { eventBus.off('notification') // 或者取消所有监听 // eventBus.all.clear() }) </script>

2.在多个组件中使用

vue

<!-- Header.vue --> <script setup lang="ts"> import eventBus from '@/utils/eventBus' const logout = () => { eventBus.emit('user-logout') } const sendNotification = () => { eventBus.emit('notification', '新消息!') } </script>

vue

<!-- Sidebar.vue --> <script setup lang="ts"> import { ref, onMounted, onUnmounted } from 'vue' import eventBus from '@/utils/eventBus' const notification = ref('') const handleNotification = (message: string) => { notification.value = message setTimeout(() => notification.value = '', 3000) } onMounted(() => { eventBus.on('notification', handleNotification) }) onUnmounted(() => { eventBus.off('notification', handleNotification) }) </script>

高级用法

1.使用工厂模式创建多个事件总线

// src/utils/eventBusFactory.ts import mitt from 'mitt' export function createEventBus<T>() { return mitt<T>() } // 创建不同的事件总线 export const uiEventBus = createEventBus<{ 'modal-open': { id: string } 'modal-close': string }>() export const dataEventBus = createEventBus<{ 'data-loaded': any[] 'data-error': Error }>()

2.封装为 Vue 插件

// src/plugins/eventBus.ts import { type App } from 'vue' import mitt, { type Emitter } from 'mitt' // 事件类型定义 type Events = { [key: string]: any } // 创建全局事件总线 const eventBus: Emitter<Events> = mitt<Events>() export const EventBusPlugin = { install(app: App) { // 全局属性 app.config.globalProperties.$eventBus = eventBus // 提供/注入 app.provide('eventBus', eventBus) } } // 在 Composition API 中使用的 Hook export function useEventBus() { const eventBus = inject<Emitter<Events>>('eventBus') if (!eventBus) { throw new Error('Event bus not provided') } return eventBus } export default eventBus

main.ts

// main.ts import { createApp } from 'vue' import { EventBusPlugin } from '@/plugins/eventBus' import App from './App.vue' const app = createApp(App) app.use(EventBusPlugin) app.mount('#app')

3.在 Composition API 中封装 Hook

typescript

// src/composables/useEventBus.ts import { onUnmounted } from 'vue' import eventBus, { type Handler } from '@/utils/eventBus' export function useEventBus() { const listeners: Array<[string, Handler]> = [] const emit = <T = any>(event: string, payload?: T) => { eventBus.emit(event, payload) } const on = <T = any>(event: string, handler: (payload: T) => void) => { eventBus.on(event, handler as Handler) listeners.push([event, handler as Handler]) } const off = <T = any>(event: string, handler: (payload: T) => void) => { eventBus.off(event, handler as Handler) const index = listeners.findIndex( ([e, h]) => e === event && h === handler ) if (index > -1) { listeners.splice(index, 1) } } const once = <T = any>(event: string, handler: (payload: T) => void) => { const onceHandler = (payload: T) => { handler(payload) off(event, onceHandler) } on(event, onceHandler) } // 自动清理监听器 onUnmounted(() => { listeners.forEach(([event, handler]) => { eventBus.off(event, handler) }) listeners.length = 0 }) return { emit, on, off, once } }

vue

<!-- 使用封装的 Hook --> <script setup lang="ts"> import { useEventBus } from '@/composables/useEventBus' const { emit, on, once } = useEventBus() // 发送事件 const sendEvent = () => { emit('custom-event', { data: 'test' }) } // 监听事件 on('custom-event', (payload) => { console.log('收到事件:', payload) }) // 只监听一次 once('one-time-event', (payload) => { console.log('只会触发一次:', payload) }) </script>

与 Vue 原生方法的比较

特性MittVue 3 的emit/propsProvide/Inject
通信范围任意组件间父子组件间祖先-后代组件间
类型支持TypeScript 友好TypeScript 友好TypeScript 友好
耦合度
适用场景全局事件、兄弟组件父子组件层级深的组件

最佳实践

  1. 类型安全

    // 正确定义事件类型 type Events = { 'user-updated': User 'cart-changed': CartItem[] }
  2. 及时清理

    // 组件卸载时取消监听 onUnmounted(() => { eventBus.off('event-name', handler) })
  3. 避免过度使用

    • 优先使用 props/emit 进行父子组件通信

    • 优先使用 provide/inject 进行层级通信

    • 只在需要跨多级/兄弟组件通信时使用事件总线

  4. 错误处理

    const { emit, on } = useEventBus() on('error-event', (error) => { // 统一错误处理 console.error('事件错误:', error) })

注意事项

  1. 内存泄漏:务必在组件卸载时取消事件监听

  2. 调试困难:事件总线可能导致数据流不清晰

  3. 替代方案对于复杂应用,考虑使用 Pinia 进行状态管理


Mitt 在 Vue 3 中是一个简单有效的跨组件通信方案,但应谨慎使用,避免滥用导致代码难以维护。

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/4/18 3:38:34

BiliTools智能解析工具:视频内容提取的终极指南

BiliTools智能解析工具&#xff1a;视频内容提取的终极指南 【免费下载链接】BiliTools A cross-platform bilibili toolbox. 跨平台哔哩哔哩工具箱&#xff0c;支持视频、音乐、番剧、课程下载……持续更新 项目地址: https://gitcode.com/GitHub_Trending/bilit/BiliTools …

作者头像 李华
网站建设 2026/4/18 3:36:25

Qwen All-in-One支持哪些硬件?CPU兼容性测试报告

Qwen All-in-One支持哪些硬件&#xff1f;CPU兼容性测试报告 1. &#x1f9e0; Qwen All-in-One: 单模型多任务智能引擎 基于 Qwen1.5-0.5B 的轻量级、全能型 AI 服务 Single Model, Multi-Task Inference powered by LLM Prompt Engineering 你有没有遇到过这种情况&#xff…

作者头像 李华
网站建设 2026/4/18 3:37:39

OpCore Simplify智能配置:零基础打造完美黑苹果的终极指南

OpCore Simplify智能配置&#xff1a;零基础打造完美黑苹果的终极指南 【免费下载链接】OpCore-Simplify A tool designed to simplify the creation of OpenCore EFI 项目地址: https://gitcode.com/GitHub_Trending/op/OpCore-Simplify 传统黑苹果配置过程充满了技术挑…

作者头像 李华
网站建设 2026/4/17 20:31:44

学校事务管理系统。 系统基于 vue+springboot+mybatisplus 开发的前后...

学校事务管理系统。 系统基于 vuespringbootmybatisplus 开发的前后台分离项目。 系统亮点&#xff1a;角色权限的设置&#xff0c;分6中不同角色。 超级管理员功能&#xff1a;角色管理&#xff0c;权限管理&#xff08;给角色分配菜单权限&#xff09;&#xff0c;管理员设置…

作者头像 李华
网站建设 2026/4/18 3:35:50

RTL8812AU无线网卡驱动完全配置指南:从基础安装到高级功能

RTL8812AU无线网卡驱动完全配置指南&#xff1a;从基础安装到高级功能 【免费下载链接】rtl8812au RTL8812AU/21AU and RTL8814AU driver with monitor mode and frame injection 项目地址: https://gitcode.com/gh_mirrors/rt/rtl8812au 想要充分发挥你的RTL8812AU、RT…

作者头像 李华
网站建设 2026/4/18 3:38:34

OpCore Simplify完整指南:从零开始构建黑苹果系统的智能解决方案

OpCore Simplify完整指南&#xff1a;从零开始构建黑苹果系统的智能解决方案 【免费下载链接】OpCore-Simplify A tool designed to simplify the creation of OpenCore EFI 项目地址: https://gitcode.com/GitHub_Trending/op/OpCore-Simplify 传统黑苹果配置过程复杂且…

作者头像 李华