手写发布订阅模式

手写实现一个简易的发布订阅模式,通常有以下几个关键点:

  1. 订阅(subscribe):用户订阅特定的事件,当该事件触发时,执行与事件关联的回调函数。

  2. 发布(publish):当某个事件发生时,发布该事件,并通知所有订阅了该事件的回调函数。

发布订阅模式实现

我们可以创建一个 EventEmitter 类,实现 subscribe、unsubscribe(取消订阅)和 publish(发布)方法。

实现代码:

class EventEmitter {

constructor() {

// 存储事件名和对应的订阅者列表

this.events = {};

}

// 订阅事件

subscribe(event, listener) {

if (!this.eventsevent) {

this.eventsevent = \[\];

}

this.eventsevent.push(listener); // 将订阅者(回调函数)添加到事件的订阅者列表

return () => this.unsubscribe(event, listener); // 返回取消订阅的函数

}

// 取消订阅

unsubscribe(event, listener) {

const listeners = this.eventsevent;

if (!listeners) return;

复制代码
const index = listeners.indexOf(listener);
if (index !== -1) {
  listeners.splice(index, 1);  // 移除指定的回调函数
}

}

// 发布事件

publish(event, ...args) {

const listeners = this.eventsevent;

if (!listeners) return;

复制代码
listeners.forEach(listener => listener(...args));  // 执行每个订阅者的回调

}

}

说明:

  1. subscribe:

如果事件尚未存在,则为该事件创建一个空数组。

将订阅者(即回调函数)推入事件对应的数组中。

返回一个取消订阅的函数,方便用户手动移除某个事件的订阅。

  1. unsubscribe:

用于取消订阅,查找指定事件的回调函数并从订阅者列表中移除它。

  1. publish:

用于触发事件,通知所有订阅者。

通过 forEach 遍历事件对应的回调列表,执行每个回调函数。

使用示例:

const emitter = new EventEmitter();

// 订阅事件

const onUserLogin = (username) => {

console.log(${username} has logged in!);

};

const onUserLogout = (username) => {

console.log(${username} has logged out!);

};

const unsubscribeLogin = emitter.subscribe('login', onUserLogin);

const unsubscribeLogout = emitter.subscribe('logout', onUserLogout);

// 发布事件

emitter.publish('login', 'john_doe'); // 输出: john_doe has logged in!

emitter.publish('logout', 'john_doe'); // 输出: john_doe has logged out!

// 取消订阅

unsubscribeLogin(); // 取消登录事件的订阅

emitter.publish('login', 'jane_doe'); // 不会输出任何信息,因为已经取消订阅

// 再次发布注销事件

emitter.publish('logout', 'jane_doe'); // 输出: jane_doe has logged out!

扩展:

如果需要支持一次性事件(例如订阅者只想接收一次事件),可以实现一个 once 方法,订阅后自动取消订阅。

还可以支持事件参数的筛选、优先级等功能。

总结:

该实现提供了一个简单的发布订阅系统,支持事件的订阅、取消订阅和发布。

通过此模式,事件的发布者与订阅者之间解耦,适合用于异步通信、消息推送、事件驱动的编程等场景。

如果你需要进一步的功能扩展或者有其他问题,随时告诉我!

相关推荐
এ慕ོ冬℘゜7 分钟前
jQuery 高可用多图上传组件(企业级封装 + 踩坑全解 + 可直接上线)
前端·javascript·jquery
爱勇宝13 分钟前
AI 时代,前端工程师的话语权正在下降?
前端·后端
kymjs张涛13 分钟前
一个月,纯VibeCoding,全平台云笔记APP
前端·javascript·后端
巴勒个啦20 分钟前
esbuild 插件实战:5个真实场景带你自定义构建流水线
前端·angular.js
狗头大军之江苏分军28 分钟前
前端路由是怎么来的
前端·javascript·后端
Patrick_Wilson31 分钟前
Cookie 作用域避坑:父域泄漏、同名优先级与多环境隔离
前端·http·浏览器
api工厂38 分钟前
ZCode 3.0 版本搭配GLM-5.2能力测试
前端·人工智能·ai
小小小小宇42 分钟前
单点登录(二)
前端
阿猫的故乡1 小时前
Vue + Axios 从入门到封装:拦截器、错误处理、请求取消、接口管理全搞定
前端·javascript·vue.js
良逍Ai出海1 小时前
免费模板搭完独立站后,我用 Codex + Figma 做了自己的页面设计
前端·人工智能·figma