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

news2025/6/11 14:31:30

解锁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 = newAge
    console.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.coloradmin.cn/o/2406284.html

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

相关文章

实现p2p的webrtc-srs版本

1. 基本知识 1.1 webrtc 一、WebRTC的本质:实时通信的“网络协议栈”类比 将WebRTC类比为Linux网络协议栈极具洞察力,二者在架构设计和功能定位上高度相似: 分层协议栈架构 Linux网络协议栈:从底层物理层到应用层(如…

第2篇:BLE 广播与扫描机制详解

本文是《BLE 协议从入门到专家》专栏第二篇,专注于解析 BLE 广播(Advertising)与扫描(Scanning)机制。我们将从协议层结构、广播包格式、设备发现流程、控制器行为、开发者 API、广播冲突与多设备调度等方面,全面拆解这一 BLE 最基础也是最关键的通信机制。 一、什么是 B…

开源 vGPU 方案:HAMi,实现细粒度 GPU 切分

本文主要分享一个开源的 GPU 虚拟化方案:HAMi,包括如何安装、配置以及使用。 相比于上一篇分享的 TimeSlicing 方案,HAMi 除了 GPU 共享之外还可以实现 GPU core、memory 得限制,保证共享同一 GPU 的各个 Pod 都能拿到足够的资源。…

盲盒一番赏小程序:引领盲盒新潮流

在盲盒市场日益火爆的今天,如何才能在众多盲盒产品中脱颖而出?盲盒一番赏小程序给出了答案,它以创新的玩法和优质的服务,引领着盲盒新潮流。 一番赏小程序的最大特色在于其独特的赏品分级制度。赏品分为多个等级,从普…

边缘计算设备全解析:边缘盒子在各大行业的落地应用场景

随着工业物联网、AI、5G的发展,数据量呈爆炸式增长。但你有没有想过,我们生成的数据,真的都要发回云端处理吗?其实不一定。特别是在一些对响应时间、网络带宽、数据隐私要求高的行业里,边缘计算开始“火”了起来&#…

Linux实现线程同步的方式有哪些?

什么是线程同步? 想象一下超市收银台:如果所有顾客(线程)同时挤向同一个收银台(共享资源),场面会一片混乱。线程同步就是给顾客们发"排队号码牌",确保: 有序访…

python学习day39

图像数据与显存 知识点回顾 1.图像数据的格式:灰度和彩色数据 2.模型的定义 3.显存占用的4种地方 a.模型参数梯度参数 b.优化器参数 c.数据批量所占显存 d.神经元输出中间状态 4.batchisize和训练的关系 import torch import torchvision import torch.nn as nn imp…

年度峰会上,抖音依靠人工智能和搜索功能吸引广告主

上周早些时候举行的第五届年度TikTok World产品峰会上,TikTok推出了一系列旨在增强该应用对广告主吸引力的功能。 新产品列表的首位是TikTok Market Scope,这是一个全新的分析平台,为广告主提供整个考虑漏斗的全面视图,使他们能够…

如何使用CodeRider插件在IDEA中生成代码

一、环境搭建与插件安装 1.1 环境准备 名称要求说明操作系统Windows 11JetBrains IDEIntelliJ IDEA 2025.1.1.1 (Community Edition)硬件配置推荐16GB内存50GB磁盘空间 1.2 插件安装流程 步骤1:市场安装 打开IDEA,进入File → Settings → Plugins搜…

电脑定时关机工具推荐

软件介绍 本文介绍一款轻量级的电脑自动关机工具,无需安装,使用简单,可满足定时关机需求。 工具简介 这款关机助手是一款无需安装的小型软件,文件体积仅60KB,下载后可直接运行,无需复杂配置。 使用…

前端异步编程全场景解读

前端异步编程是现代Web开发的核心,它解决了浏览器单线程执行带来的UI阻塞问题。以下从多个维度进行深度解析: 一、异步编程的核心概念 JavaScript的执行环境是单线程的,这意味着在同一时间只能执行一个任务。为了不阻塞主线程,J…

分布式光纤声振传感技术原理与瑞利散射机制解析

分布式光纤传感技术(Distributed Fiber Optic Sensing,简称DFOS)作为近年来迅速发展的新型感知手段,已广泛应用于边界安防、油气管道监测、结构健康诊断、地震探测等领域。其子类技术——分布式光纤声振传感(Distribut…

RocketMQ 客户端负载均衡机制详解及最佳实践

延伸阅读:🔍「RocketMQ 中文社区」 持续更新源码解析/最佳实践,提供 RocketMQ 专家 AI 答疑服务 前言 本文介绍 RocketMQ 负载均衡机制,主要涉及负载均衡发生的时机、客户端负载均衡对消费的影响(消息堆积/消费毛刺等…

【SSM】SpringMVC学习笔记7:前后端数据传输协议和异常处理

这篇学习笔记是Spring系列笔记的第7篇,该笔记是笔者在学习黑马程序员SSM框架教程课程期间的笔记,供自己和他人参考。 Spring学习笔记目录 笔记1:【SSM】Spring基础: IoC配置学习笔记-CSDN博客 对应黑马课程P1~P20的内容。 笔记2…

C++课设:实现本地留言板系统(支持留言、搜索、标签、加密等)

名人说:路漫漫其修远兮,吾将上下而求索。—— 屈原《离骚》 创作者:Code_流苏(CSDN)(一个喜欢古诗词和编程的Coder😊) 专栏介绍:《编程项目实战》 目录 一、项目功能概览与亮点分析1. 核心功能…

【见合八方平面波导外腔激光器专题系列】用于干涉光纤传感的低噪声平面波导外腔激光器2

----翻译自Mazin Alalus等人的文章 摘要 1550 nm DWDM 平面波导外腔激光器具有低相位/频率噪声、窄线宽和低 RIN 等特点。该腔体包括一个半导体增益芯片和一个带布拉格光栅的平面光波电路波导,采用 14 引脚蝶形封装。这种平面波导外腔激光器设计用于在振动和恶劣的…

Xcode 16.2 版本 pod init 报错

Xcode 版本升级到 16.2 后,项目执行 pod init 报错; ### Error RuntimeError - PBXGroup attempted to initialize an object with unknown ISA PBXFileSystemSynchronizedRootGroup from attributes: {"isa">"PBXFileSystemSynchron…

timestamp时间戳转换工具

作为一名程序员,一款高效的 在线转换工具 (在线时间戳转换 计算器 字节单位转换 json格式化)必不可少!https://jsons.top 排查问题时非常痛的点: 经常在秒级、毫秒级、字符串格式的时间单位来回转换,于是决定手撸一个…

数据库管理与高可用-MySQL故障排查与生产环境优化

目录 #1.1MySQL单案例故障排查 1.1.1MySQL常见的故障排查 1.1.2MySQL主从故障排查 #2.1MySQL优化 2.1.1硬件方面的优化 2.1.2进程方面的优化 #3.1MySQL存储引擎 3.1.1 MyISAM存储引擎 3.1.2 InnoDB存储引擎 1.1MySQL单案例故障排查 1.1.1MySQL常见的故障排查 (1&…

LangChain + LangSmith + DeepSeek 入门实战:构建代码生成助手

本文基于 Jupyter Notebook 实践代码,结合 LangChain、LangSmith 和 DeepSeek 大模型,手把手演示如何构建一个代码生成助手,并实现全流程追踪与优化。 一、环境准备与配置 1. 安装依赖 pip install langchain langchain_openai2. 设置环境变…