Mitt 事件发射器完全指南:200字节的轻量级解决方案

简介

Mitt 是一个轻量级的事件发射器库,体积小巧(约 200 字节),无依赖,支持 TypeScript。它提供了简单而强大的事件发布/订阅机制,适用于组件间通信、状态管理等场景。

特点

  • 🚀 超轻量级(~200 bytes)
  • 🔧 无依赖
  • 📝 TypeScript 支持
  • 🎯 简单易用的 API
  • 🛡️ 类型安全

安装

复制代码
npm install mitt

基本用法

创建事件发射器

复制代码
import mitt from "mitt";

// 创建事件发射器实例
const emitter = mitt();

// 或者指定事件类型
type Events = {
  foo: string;
  bar: number;
  baz: void;
};

const typedEmitter = mitt<Events>();

监听事件

复制代码
// 监听单个事件
emitter.on("foo", (data) => {
  console.log("收到 foo 事件:", data);
});

// 监听多个事件
emitter.on("bar", (data) => {
  console.log("收到 bar 事件:", data);
});

// 使用通配符监听所有事件
emitter.on("*", (type, data) => {
  console.log(`事件类型: ${type}, 数据:`, data);
});

发射事件

复制代码
// 发射事件(无数据)
emitter.emit("baz");

// 发射事件(带数据)
emitter.emit("foo", "hello world");
emitter.emit("bar", 42);

移除监听器

复制代码
// 移除特定事件的监听器
emitter.off("foo");

// 移除所有监听器
emitter.all.clear();

// 移除特定监听器(需要保存引用)
const handler = (data: string) => console.log(data);
emitter.on("foo", handler);
emitter.off("foo", handler);

API 详解

mitt()

创建一个新的事件发射器实例。

复制代码
const emitter = mitt<Events>();

emitter.on(type, handler)

注册事件监听器。

参数:

  • type: 事件类型(字符串或通配符 '*')
  • handler: 事件处理函数

返回值:

复制代码
emitter.on("message", (data: string) => {
  console.log("收到消息:", data);
});

emitter.off(type, handler?)

移除事件监听器。

参数:

  • type: 事件类型
  • handler: 可选的特定处理函数

返回值:

复制代码
// 移除所有 'message' 事件的监听器
emitter.off("message");

// 移除特定的监听器
const handler = (data: string) => console.log(data);
emitter.off("message", handler);

emitter.emit(type, data?)

发射事件。

参数:

  • type: 事件类型
  • data: 可选的事件数据

返回值:

复制代码
emitter.emit("message", "Hello World");
emitter.emit("notification");

emitter.all

获取所有注册的事件监听器。

复制代码
// 查看所有事件类型
console.log(Object.keys(emitter.all));

// 清空所有监听器
emitter.all.clear();

高级特性

通配符监听

复制代码
// 监听所有事件
emitter.on("*", (type, data) => {
  console.log(`事件 ${type} 被触发:`, data);
});

// 通配符监听器会收到事件类型作为第一个参数
emitter.emit("foo", "bar");
// 输出: 事件 foo 被触发: bar

链式调用

复制代码
// 支持链式调用
emitter
  .on("event1", () => console.log("event1"))
  .on("event2", () => console.log("event2"))
  .emit("event1");

常见用例

React 组件间通信

复制代码
// 创建全局事件发射器
export const globalEmitter = mitt<{
  themeChange: "light" | "dark";
  userUpdate: { id: string; name: string };
}>();

// 在组件中使用
function ThemeToggle() {
  const toggleTheme = () => {
    globalEmitter.emit("themeChange", "dark");
  };

  return <button onClick={toggleTheme}>切换主题</button>;
}

function UserProfile() {
  useEffect(() => {
    const handler = (user: { id: string; name: string }) => {
      console.log("用户信息更新:", user);
    };

    globalEmitter.on("userUpdate", handler);
    return () => globalEmitter.off("userUpdate", handler);
  }, []);

  return <div>用户资料</div>;
}

Vue 3 组合式 API

复制代码
// composables/useEventBus.ts
import mitt from "mitt";
import { onUnmounted } from "vue";

export const eventBus = mitt();

export function useEventBus() {
  const handlers: Array<() => void> = [];

  const on = (event: string, handler: Function) => {
    eventBus.on(event, handler);
    handlers.push(() => eventBus.off(event, handler));
  };

  const emit = (event: string, data?: any) => {
    eventBus.emit(event, data);
  };

  onUnmounted(() => {
    handlers.forEach((cleanup) => cleanup());
  });

  return { on, emit };
}

// 在组件中使用
export default {
  setup() {
    const { on, emit } = useEventBus();

    on("message", (data: string) => {
      console.log("收到消息:", data);
    });

    const sendMessage = () => {
      emit("message", "Hello from component");
    };

    return { sendMessage };
  },
};

注意事项

内存泄漏

  • 始终在组件卸载时清理事件监听器
  • 使用 emitter.all.clear() 清空所有监听器
  • 保存处理函数引用以便精确移除

事件循环

  • 避免在事件处理函数中发射相同事件
  • 使用防抖或节流处理高频事件
  • 考虑使用 setTimeoutsetImmediate 延迟事件发射

调试技巧

复制代码
// 开发环境下的调试辅助
if (process.env.NODE_ENV === "development") {
  const originalEmit = emitter.emit;
  emitter.emit = (type, data) => {
    console.log(`[Event] ${type}:`, data);
    originalEmit.call(emitter, type, data);
  };
}

Mitt 是一个简单而强大的事件发射器,通过合理使用可以大大简化组件间通信和状态管理。掌握这些用法和最佳实践,将帮助您构建更加健壮和可维护的应用程序。

Mitt 事件发射器完全指南:200字节的轻量级解决方案 - 高质量源码分享平台-免费下载各类网站源码与模板及前沿技术分享

相关推荐
Highcharts.js2 小时前
Highcharts React v4 迁移指南(下):分步代码示例与常见问题解决
javascript·react.js·typescript·react·highcharts·代码示例·v4迁移
前端Hardy2 小时前
别再手动写 loading 了!封装一个自动防重提交的 Hook
前端·javascript·vue.js
前端Hardy2 小时前
前端如何实现“无感刷新”Token?90% 的人都做错了
前端·javascript·vue.js
SuperEugene3 小时前
Vue Router 实战规范:path/name/meta 配置 + 动态 / 嵌套路由,统一团队标准|状态管理与路由规范篇
开发语言·前端·javascript·vue.js·前端框架
小彭努力中3 小时前
194.Vue3 + OpenLayers 实战:动态位置 + 高度 + 角度,模拟卫星地面覆盖范围
前端·css·vue.js·openlayers·animate
张一凡933 小时前
easy-model -- "小而美"的React状态管理方案
前端·javascript·react.js
前端Hardy3 小时前
纯 HTML/CSS/JS 实现的高颜值登录页,还会眨眼睛!少女心爆棚!
前端·javascript·vue.js
miss4 小时前
Vue2 → Vue3 深度对比:8 大核心优化,性能提升 2 倍
前端·vue.js·架构
重铸码农荣光4 小时前
手写一个精简版 Zustand:深入理解 React 状态管理的核心原理
react.js·面试·源码
angerdream4 小时前
最新版vue3+TypeScript开发入门到实战教程之生命周期函数
javascript·vue.js