手写发布-订阅模式

简介: 发布-订阅模式其实是一种对象间一对多的依赖关系,当一个对象的状态发送改变时,所有依赖于它的对象都将得到状态改变的通知。

一、 什么是发布-订阅模式

1. 定义

发布-订阅模式其实是一种对象间一对多的依赖关系,当一个对象的状态发送改变时,所有依赖于它的对象都将得到状态改变的通知。

订阅者(Subscriber)把自己想订阅的事件注册(Subscribe)到调度中心(Event Channel),当发布者(Publisher)发布该事件(Publish Event)到调度中心,也就是该事件触发时,由调度中心统一调度(Fire Event)订阅者注册到调度中心的处理代码。

2. 例子

我们关注的公众号文章,给我们进行推送时候,即是发布订阅模式

公众号属于发布者,我们属于订阅者

用户将订阅公众号的事件注册到调度中心,公众号作为发布者,当有新文章发布时,公众号发布该事件到调度中心,调度中心会及时发消息告知用户。

二、 如何实现发布-订阅模式?

1. 实现思路

  • 创建一个对象
  • 在该对象上创建一个调度中心
  • on 方法用来把函数 fn 都加到缓存列表中(订阅者注册事件到调度中心)
  • emit 方法取到 arguments 里第一个当做 event,根据 event 值去执行对应缓存列表中的函数(发布者发布事件到调度中心,调度中心处理代码)
  • off 方法可以根据 event 值取消订阅(取消订阅)
  • once 方法只监听一次,调用完毕后删除缓存函数(订阅一次)

2. 代码

// 公众号调度中心
class EventHub {
  constructor(){
    // 缓存列表,存放 event 及 fn
    this.map = {};
  }
  on(event, fn) {
     // 如果对象中没有对应的 event 值,也就是说明没有订阅过,就给 event 创建个缓存列表
    this.map[event] = this.map[event] || []
    // 把 fn 添加到对应 event 的缓存列表里
    this.map[event].push(fn)
  }
  emit(event, data) {
    const fnList = this.map[event] || []
    // 如果缓存列表里没有 fn 就return
    if (!fnList || fnList.length === 0) return;
    // 遍历 event 值对应的缓存列表,依次执行 fn
    fnList.forEach(fn => fn.call(undefined, data))
  }
  off(event, fn) {
    const fnList = this.map[event] || []
    const index = fnList.indexOf(fn)
    // 如果缓存列表里没有要找的fn 就return
    if(index < 0) return
    fnList.splice(index, 1)
  }
  //执行一次
  once(event, callback){
    const f = (data) => {
      callback(data);    
      this.off(event, f);
    }
    this.on(event, f);
  }
}
function user1 (content) {
    console.log('用户1订阅了:', content);
};
function user2 (content) {
    console.log('用户2订阅了:', content);
};
function user3 (content) {
    console.log('用户3订阅了:', content);
}
function user4 (content) {
    console.log('用户4订阅了:', content);
}
const e = new EventHub()
// 订阅
e.on('article1', user1);
e.on('article1', user2);
e.on('article1', user3);
// 取消user2方法的订阅
e.off('article1', user2);
e.once('article2', user4)
// 发布
e.emit('article1', 'Javascript 发布-订阅模式');
e.emit('article1', 'Javascript 发布-订阅模式');
e.emit('article2', 'Javascript 观察者模式');
e.emit('article2', 'Javascript 观察者模式');
/*
    用户1订阅了: Javascript 发布-订阅模式
    用户3订阅了: Javascript 发布-订阅模式
    用户1订阅了: Javascript 发布-订阅模式
    用户3订阅了: Javascript 发布-订阅模式
    用户4订阅了: Javascript 观察者模式
*/

三、 总结

1. 优点

  • 对象之间解耦
  • 异步编程中,可以更松耦合的代码编写

2. 缺点

  • 创建订阅者本身要消耗一定的时间和内存
  • 虽然可以弱化对象之间的联系,多个发布者和订阅者嵌套一起的时候,程序难以跟踪维护



目录
相关文章
|
3月前
|
设计模式 消息中间件 安全
C# 一分钟浅谈:观察者模式与订阅发布模式
【10月更文挑战第11天】本文介绍了观察者模式和订阅发布模式,这两种设计模式主要用于实现对象间的解耦。观察者模式通过事件和委托实现一个对象状态改变时通知多个依赖对象;订阅发布模式则通过事件聚合器实现发布者与订阅者之间的解耦。文章详细探讨了这两种模式的实现方式、常见问题及避免方法,帮助开发者在实际项目中更好地应用这些模式,提升代码的可维护性和扩展性。
85 1
|
2月前
|
测试技术
发布模式
蓝绿部署是一种应用发布技术,旨在减少发布时的中断时间和快速回滚。它通过维护两个几乎相同的生产环境(蓝绿系统)来实现,其中一个处于活动状态(绿色),另一个待命(蓝色)。新版本先在蓝色环境中测试,确认无误后切换用户至该环境,若发现问题可迅速回滚至绿色环境。此策略适用于内聚性较强的系统,对于复杂系统则需额外考量数据同步等问题。
30 0
|
7月前
|
存储 负载均衡 安全
中间件消息发布-订阅模式
【6月更文挑战第9天】
154 5
|
设计模式 Go
Go语言事件系统设计解析:发布-订阅模式实战
Go语言事件系统设计解析:发布-订阅模式实战
522 0
|
8月前
|
设计模式 负载均衡 算法
C/C++发布-订阅者模式世界:揭秘高效编程的秘诀
C/C++发布-订阅者模式世界:揭秘高效编程的秘诀
348 1
|
8月前
|
移动开发 小程序 Go
【社区每周】小程序消息订阅插件升级为消息订阅接口(2022年8月第五期)
【社区每周】小程序消息订阅插件升级为消息订阅接口(2022年8月第五期)
57 0
|
小程序
小程序订阅消息推送简要流程图
小程序订阅消息推送简要流程图
129 0
小程序订阅消息推送简要流程图
|
SQL 消息中间件 Java
想要流畅体验 TDengine 3.0 数据订阅功能?要点都在这里
在本文中,TDengine 资深研发将以 TDengine 3.0 为对象,为大家介绍数据订阅功能的正确打开方式,给到有需要的人作参考指南,避免走入应用误区。
234 0
|
设计模式 开发者
设计模式之订阅发布模式
设计模式之订阅发布模式
253 0
|
Prometheus Kubernetes Cloud Native
Flagger(应用自动发布)介绍和原理剖析
## 简介 [Flagger](https://github.com/weaveworks/flagger)是一个能使运行在k8s体系上的应用发布流程全自动(无人参与)的工具, 它能减少发布的人为关注时间, 并且在发布过程中能自动识别一些风险(例如:RT,成功率,自定义metrics)并回滚. ## 主要特性 ![features](https://intranetproxy.ali
4542 0