JS设计模式(5): 发布订阅模式

解锁JavaScript发布订阅模式:让代码沟通更优雅

在JavaScript的世界里,我们常常会遇到这样的场景:多个模块之间需要相互通信,但是又不想让它们产生过于紧密的耦合。这时候,发布订阅模式就像一位优雅的信使,能够帮助我们实现模块间的解耦,让代码的结构更加清晰、可维护。今天,我们就来深入探讨一下JavaScript中的发布订阅模式,并用一段代码来揭开它的神秘面纱。

什么是发布订阅模式?

发布订阅模式(Publish/Subscribe Pattern)是一种消息范式,发布者(Publisher)不会将消息直接发送给特定的订阅者(Subscriber),而是将发布的消息分为不同的类别,无需了解哪些订阅者(如果有的话)可能存在。同样的,订阅者可以表达对一个或多个类别的兴趣,只接收感兴趣的消息,无需了解哪些发布者(如果有的话)存在。这种模式实现了发布者和订阅者之间的解耦,使得它们可以独立变化而互不影响。

在JavaScript的编程场景中,比如我们开发一个复杂的Web应用,有多个组件需要响应某个事件的变化,例如用户登录状态的改变、数据的更新等。如果采用传统的方式,组件之间可能需要相互引用、层层调用,这会导致代码的耦合度极高,后期维护和扩展都非常困难。而发布订阅模式则提供了一种更好的解决方案,它允许组件之间通过事件进行通信,组件只需要关心自己感兴趣的事件,而无需了解事件的来源和其他组件的内部实现。

发布订阅模式的实现

下面,我们通过一段代码来实现一个简单的发布订阅模式。我们创建一个EventEmitter类,它将作为我们的事件发射器,负责管理事件的订阅、发布和取消订阅等操作。

class EventEmitter {constructor () {this.event = {}}// 订阅事件on (eventName, callback) {if (!this.event[eventName]) {this.event[eventName] = []}this.event[eventName].push(callback)return this}// 发布事件emit (eventName, ...args) {if (this.event[eventName]) {this.event[eventName].forEach(callback => callback(...args))}return this}// 取消订阅off (eventName, callback) {if (this.event[eventName]) {this.event[eventName] = this.event[eventName].filter(cb => cb!== callback)}return this}// 单次订阅once (eventName, callback) {const wrapper = (...args) => {callback(...args)this.off(eventName, wrapper)}this.on(eventName, wrapper)}
}

在上述代码中,EventEmitter类的构造函数初始化了一个空对象event,用于存储不同事件名称对应的回调函数数组。on方法用于订阅事件,它会将传入的回调函数添加到对应事件名称的数组中。emit方法用于发布事件,当调用emit时,它会查找对应事件名称的回调函数数组,并依次执行数组中的每个回调函数,同时将传入的参数传递给回调函数。off方法用于取消订阅,它会从对应事件名称的回调函数数组中过滤掉指定的回调函数。once方法用于实现单次订阅,即回调函数只会执行一次,执行完毕后会自动取消订阅。

发布订阅模式的实际应用示例

为了更好地理解发布订阅模式的实际应用,我们来看一个具体的例子。假设我们有一个表示人的对象person,我们想要在其age属性发生变化时,通知其他相关的对象进行相应的处理。我们可以使用Proxy结合发布订阅模式来实现这个功能。

const eventBus = new EventEmitter()const obj = {age: 21,onAgeChange: function (newAge) {this.age = newAgeconsole.log('age 发生改变,现在是', this.age)}
}const person = {age: 10,
}const proxyPerson = new Proxy(person, {set: (target, key, newValue) => {eventBus.emit(`${key}Change`, newValue)}
})eventBus.on('ageChange', obj.onAgeChange)proxyPerson.age = 20
proxyPerson.age = 100

在这个例子中,我们首先创建了一个EventEmitter实例eventBus作为事件总线。然后,我们定义了一个obj对象,它包含一个age属性和一个onAgeChange方法,用于处理age属性变化的逻辑。接着,我们创建了一个person对象,并使用Proxy对其进行代理。在Proxyset方法中,当person对象的属性发生变化时,我们通过eventBus发布一个ageChange事件,并将新的值作为参数传递出去。最后,我们通过eventBuson方法将objonAgeChange方法订阅到ageChange事件上。当我们修改proxyPersonage属性时,就会触发ageChange事件,objonAgeChange方法也会被调用,从而实现了age属性变化的通知和处理。

发布订阅模式的优势与注意事项

优势

  1. 解耦:发布者和订阅者之间没有直接的依赖关系,它们只通过事件进行通信。这样,当我们需要添加、删除或修改某个组件时,不会影响到其他组件,大大提高了代码的可维护性和扩展性。
  2. 灵活性:可以很方便地添加新的订阅者或发布者,而不需要修改已有的代码。例如,在我们的例子中,如果后续还有其他对象需要响应age属性的变化,只需要通过eventBus订阅ageChange事件即可。
  3. 可复用性EventEmitter类可以在多个项目或模块中复用,减少了重复代码的编写。

注意事项

  1. 内存管理:如果订阅者订阅了大量的事件,并且没有及时取消订阅,可能会导致内存泄漏。因此,在使用发布订阅模式时,一定要注意及时取消不再需要的订阅。
  2. 调试困难:由于事件的发布和订阅是异步进行的,而且可能涉及多个组件之间的通信,所以在调试时可能会比较困难。我们可以通过添加日志输出等方式来辅助调试。

总结

发布订阅模式是JavaScript中一种非常实用的设计模式,它通过解耦组件之间的通信,让我们的代码更加灵活、可维护和可扩展。通过本文的介绍和示例,相信你对发布订阅模式已经有了更深入的理解。在今后的JavaScript开发中,不妨尝试运用发布订阅模式,让你的代码沟通更加优雅!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.pswp.cn/diannao/86619.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【电路物联网】SDN架构与工作原理介绍

(꒪ꇴ꒪ ),Hello我是祐言QAQ我的博客主页:C/C语言,数据结构,Linux基础,ARM开发板,网络编程等领域UP🌍快上🚘,一起学习,让我们成为一个强大的攻城狮&#xff0…

vscode 保存 js 时会自动格式化,取消设置也不好使

vscode 里的设置搜索 Editor: Format On Save 取消勾选 卸载 Prettier - Code formatter 这个插件后好使了,本来以为是插件的问题,后来发现是工作区设置的问题。 因为我是用 GitHub 下载的工程打开后, vscode 认为是工作区了, 因为 .vscode…

xcode中project.pbxproj点开为空白问题

由于需要修改signing里面的配置,点击了project.pbxproj。但是发现一片空白,如图 以为是配置文件损坏,邮件show in Finder看了一通后没看出什么所以然。并且发现entitlement文件、list文件全都是点开为白,并且没有任何保存 最后发…

JUC并发编程(四)常见模式

目录 一 同步与协调模式 1 保护性暂停模式 2 顺序控制模式 3 生产者消费者模式 4 Balking模式(犹豫模式) 二 线程管理/生命周期模式 1 两阶段终止模式 一 同步与协调模式 1 保护性暂停模式 一个线程需要等待另一个线程提供特定条件(通常是…

Vue 数据代理机制对属性名的要求

Vue 数据代理机制对属性名的要求 在 Vue 的数据代理机制中,属性名需遵循以下关键规则: 1. 禁止以 _ 或 $ 开头 ⚠️ Vue 会跳过代理以 _ 或 $ 开头的属性原因:这些前缀被 Vue 保留用于内部属性(如 _data, _uid, $refs, $el 等)示例:data() {return {count: 1, // ✅…

pdf.js在iOS移动端分页加载优化方案(ios移动端反复刷新加载问题)

背景与问题 在iOS移动端加载大型PDF文件时,由于设备内存限制,经常遇到以下问题: 内存不足导致页面崩溃大文件加载缓慢页面反复重新加载 ##解决方案 采用PDF.js的分页加载策略,实现按需加载当前可视页面及相邻页面,…

【C++】来学习使用set和map吧

各位大佬好,我是落羽!一个坚持不断学习进步的大学生。 如果您觉得我的文章有所帮助,欢迎多多互三分享交流,一起学习进步! 也欢迎关注我的blog主页: 落羽的落羽 文章目录 一、set和map是什么二、set系列1. set2. mult…

h5st逆向分析

h5st最新5.1版本逆向分析 申明定位h5st生成的位置动态插桩,事半功倍日志分析,十分钟还原算法逻辑申明 本文仅用来记录学习过程以免日后忘了,如有侵权请联系删除。 定位h5st生成的位置 通过关键字“sign”搜索,可以定位到window.PSign.sign(f)这个位置,f参数的值为{ &qu…

湖北理元理律师事务所企业债务优化路径:司法重整中的再生之道

一、企业债务危机的核心矛盾:生存与清偿的博弈 通过分析湖北理元理律师事务所经办的17件企业债务案件,发现共性难题: 债权人要求立即清偿 → 企业需持续经营造血 → 司法程序存在时间差 解决方案:构建“三重防火墙”机制 经…

链家Android面试题及参考答案

目录 请详细解释类加载的过程,包括每一步的具体实现。并说明Android中的dex分包技术及其在热更新中的应用 比较JVM和DVM的区别。在JVM中一个程序崩溃是否可能导致系统崩溃?DVM中呢? 请解释网络IP协议、TCP、UDP、HTTP、HTTPS、Socket的概念,并说明它们之间的区别 请深入…

LeetCode-多语言实现冒泡排序以及算法优化改进

目录 一、冒泡排序算法 二、应用场景/前提条件 🌈 优点 📢 缺点 三、经典算法实现并优化改进 方法一:记录最后一次交换位置,下一轮只遍历到该位置 方法二:添加标志位跟踪是否发生交换,无交换则提前终…

JAVA毕业设计227—基于SpringBoot+hadoop+spark+Vue的大数据房屋维修系统(源代码+数据库)

毕设所有选题: https://blog.csdn.net/2303_76227485/article/details/131104075 基于SpringBoothadoopsparkVue的大数据房屋维修系统(源代码数据库)227 一、系统介绍 本项目前后端分离,分为业主、维修人员、管理员三种角色 1、业主: 登…

MADlib —— 基于 SQL 的数据挖掘解决方案(9)—— 数据探索之概率统计

目录 一、概率 1. 概率的定义 2. 概率质量函数与概率密度函数 3. 条件概率 4. 期望值 二、MADlib 的概率相关函数 1. 函数语法 2. 示例 (1)求标准正态分布下,1 的概率密度函数 (2)求标准正态分布下&#xff…

耳蜗里的春天

早春的郑州飘着细雨,我牵着女儿小满的手走进市残疾人康复中心时,玻璃门内突然传来一阵清脆的笑声。穿天蓝色毛衣的小女孩戴着粉色耳蜗,正踮脚拍打着墙上的卡通贴画,银色的连接线在她耳后晃动,像一只折翼却仍在起舞的蝴…

OCR(光学字符识别)算法

OCR(光学字符识别)算法在景区护照阅读器中的应用是核心技术之一,它通过图像处理和机器学习快速提取护照信息,显著提升自动化水平。以下是其具体应用场景、技术实现及优化方向: 一、OCR在护照阅读器中的核心作用 关键信…

html打印合同模板

概述(吐槽):记录一个html打印合同模板的功能,技术栈有点杂,千禧年出产老系统的数据库是sqlserver2008,原系统框架是c#,无法二开,因为原系统的合同生成功能出现bug,没有供…

DeepCritic: SFT+RL两阶段训练突破LLM自我监督!显著提升大模型的自我批判能力!!

摘要:随着大型语言模型(LLMs)的迅速发展,对其输出进行准确反馈和可扩展监督成为一个迫切且关键的问题。利用LLMs作为批评模型以实现自动化监督是一个有前景的解决方案。在本研究中,我们专注于研究并提升LLMs在数学批评…

【深度学习】深度学习中的张量:从多维数组到智能计算单元

✅ 一、n维数组(张量,Tensor) 1. 定义 张量(Tensor)是一个通用的n维数组数据结构。 它的维度(维数)决定了它的形状,例如: 维度名称举例说明0维标量(scalar…

以太网MDI信号PCB EMC设计要点

1. PHY侧和RJ45连接器侧通用MDI布局建议 1. MDI差分对保持对称走线,走线上的焊盘封装应一致,焊盘放置位置也应对称。可以减少EMI测试中的模式转换。   2. MDI走线应保持阻抗匹配,从而减少信号线上的反射。   3. MDI走线下需有连续完整的接…

深入浅出WebGL:在浏览器中解锁3D世界的魔法钥匙

WebGL:在浏览器中解锁3D世界的魔法钥匙 引言:网页的边界正在消失 在数字化浪潮的推动下,网页早已不再是静态信息的展示窗口。如今,我们可以在浏览器中体验逼真的3D游戏、交互式数据可视化、虚拟实验室,甚至沉浸式的V…