亲宝软件园·资讯

展开

参考 EventEmitter实现一个简单的订阅发布功能函数

拾光者 人气:0

目的

订阅发布模式 应该是 JS 最常用的设计模式,也可称之为 观察者模式,前端各个方面都会涉及到它,比如:浏览器中的事件监听机制、nodejs 中的 EventEmitter 、redux 数据传递实现等等。

不管是 浏览器中的事件监听机制 还是 nodejs 中的 EventEmitter,都只能在各自的环境进行运行。因此我们需要用一份代码来让多端都可以调用,EventEmitter 功能更全面,我们借鉴它来实现一个完整的 PubSub

实现

主要参考 EventEmitter 的功能来,本篇文章提供 简单实现,下一篇提供了 完整实现,点击即可去查看,代码有问题欢迎评论区指出。

实现常用功能:

完整代码实现:

type Listener = (...args: any[]) => void;
type EventInfo = {
  // 监听器
  listener: Listener;
  // 备份,需要改变 listener 时,则需要备份,比如 once
  bak?: Listener;
};
// 创建一次性监听器
function createOnceListener(pub: PubSub, eventName: string | symbol, listener: Listener) {
  const onceListener = (...args: any[]) => {
    // 执行一次后直接取消订阅
    listener(...args);
    pub.off(eventName, listener);
  };
  return onceListener;
}
export class PubSub {
  private eventMap: Record<string | symbol, EventInfo[]> = {};
  // 订阅
  on = (eventName: string | symbol, listener: Listener) => {
    if (!this.eventMap[eventName]) {
      this.eventMap[eventName] = [];
    }
    this.eventMap[eventName].push({ listener });
    return this;
  };
  // 取消订阅
  off = (eventName: string | symbol, listener: Listener) => {
    if (this.eventMap[eventName]) {
      this.eventMap[eventName] = this.eventMap[eventName].filter((item) => {
        // once listener 取消订阅
        if (item.bak) {
          return item.bak !== listener;
        }
        // 正常 listener 取消订阅
        return item.listener !== listener;
      });
    }
    return this;
  };
  // 类似 EventEmitter 中的 emit 函数
  emit = (eventName: string | symbol, ...args: any[]) => {
    this.eventMap[eventName]?.forEach((item) => {
      item.listener(...args);
    });
    return this;
  };
  // 只订阅一次
  once = (eventName: string | symbol, listener: Listener) => {
    const onceListener = createOnceListener(this, eventName, listener);
    this.on(eventName, onceListener);
    return this;
  };
}
// 不需要有全局订阅功能的,可以去掉这个
export const pubSub = new PubSub();

实现说明:

总结

实现过程是通过阅读 EventEmitter 官方文档的 API 来进行反向实现的,typescript 类型可以直接通过 nodejs 使用的时候获取,这也一般是想实现某一个功能的方法。

有不对的地方欢迎指出,感谢

加载全部内容

相关教程
猜你喜欢
用户评论