实现发布订阅模式

发布订阅模式(Pub/Sub )是前端和后端都非常常见的一种设计模式,本质是解耦

  • 发布者(Publisher)发消息

  • 订阅者(Subscriber)接收消息

  • 中间通过一个"事件中心"来管理


🧠 一、核心思想

可以把它理解成:

"我不关心谁在听,只负责发;你不关心谁在发,只负责订阅。"

典型结构:

  • on:订阅事件

  • emit:发布事件

  • off:取消订阅


💻 二、手写一个发布订阅(JS实现)

这是最经典、面试也最常考的实现👇

复制代码
class EventBus {
  constructor() {
    this.events = {};
  }

  // 订阅
  on(eventName, callback) {
    if (!this.events[eventName]) {
      this.events[eventName] = [];
    }
    this.events[eventName].push(callback);
  }

  // 发布
  emit(eventName, ...args) {
    const handlers = this.events[eventName];
    if (handlers) {
      handlers.forEach(fn => fn(...args));
    }
  }

  // 取消订阅
  off(eventName, callback) {
    const handlers = this.events[eventName];
    if (!handlers) return;

    this.events[eventName] = handlers.filter(fn => fn !== callback);
  }

  // 只执行一次
  once(eventName, callback) {
    const fn = (...args) => {
      callback(...args);
      this.off(eventName, fn);
    };
    this.on(eventName, fn);
  }
}

🚀 三、使用示例

复制代码
const bus = new EventBus();

// 订阅
const handler = (msg) => {
  console.log('收到消息:', msg);
};

bus.on('test', handler);

// 发布
bus.emit('test', 'hello world');

// 取消订阅
bus.off('test', handler);

// 只执行一次
bus.once('onceEvent', (msg) => {
  console.log('只执行一次:', msg);
});

bus.emit('onceEvent', '第一次');
bus.emit('onceEvent', '第二次'); // 不会触发

⚔️ 四、和观察者模式的区别(面试重点)

很多人会混,这里帮你理清:

对比点 发布订阅 观察者模式
中间层 ✅ 有(EventBus) ❌ 没有
耦合度
关系 松散 直接依赖
使用场景 Vue事件、消息系统 响应式数据

👉 一句话总结:

发布订阅 = 多加了一层"中介",更解耦


🧩 五、前端实际应用场景

你项目里其实已经可能用过👇

1️⃣ Vue / React 事件通信

复制代码
// mitt / eventemitter3 本质就是 pub/sub

2️⃣ 跨组件通信(非父子)

  • EventBus

  • 全局状态管理(Redux 本质也有类似思想)

3️⃣ WebSocket 消息分发

复制代码
socket.onmessage = (msg) => {
  eventBus.emit(msg.type, msg.data);
};

4️⃣ 你之前做的性能平台(很适合用)

比如:

  • 图表联动(ECharts)

  • 筛选条件同步

  • 多模块刷新


🔥 六、进阶优化(高级面试点)

你可以再升级一下👇

✅ 1. 支持事件命名空间

复制代码
"user:login"
"user:logout"

✅ 2. 支持通配符

复制代码
"user:*"

✅ 3. 异步发布(Promise)

复制代码
emit async

✅ 4. 防止重复订阅

复制代码
Set 替代 Array

🧭 七、一句话总结

发布订阅模式 = "事件中心 + 松耦合通信",是大型前端架构必备能力

相关推荐
kyriewen5 小时前
Anthropic 估值逼近万亿美元,Claude Sonnet 5 + Claude Science 一天两连发
前端·ai编程·claude
小徐_23336 小时前
Wot UI 2.2.0 发布:Button 新增 subtle,VideoPreview 预览体验继续增强
前端·微信小程序·uni-app
山河木马7 小时前
矩阵专题3-怎么创建投影矩阵(uProjectionMatrix)
javascript·webgl·计算机图形学
天蓝色的鱼鱼8 小时前
关于 CSS 你可能不知道的属性,但关键时刻很有用
前端·css
泯泷9 小时前
第 2 篇:设计第一套字节码:Opcode、Instruction 与 Constant Pool
前端·javascript·安全
妙码生花9 小时前
从 PHP 到 AI + Golang,程序员自救转型手记(十五):优化细节、网络请求封装
前端·后端·ai编程
泯泷9 小时前
第 1 篇:从 1 + 2 开始:亲手写出第一台 JSVM
前端·javascript·安全
团团崽_七分甜9 小时前
Spring Boot 核心知识点总结
前端
lichenyang4539 小时前
从一个按钮开始,理解 ASCF 框架到底在做什么
前端