Golang观察者模式事件订阅与通知示例

Go语言通过观察者模式解耦事件发布与订阅,定义Observer和Subject接口,实现EventCenter管理订阅并通知,结合具体观察者完成状态广播。

golang观察者模式事件订阅与通知示例

在Go语言中实现观察者模式,可以很好地解耦事件的发布者与订阅者。这种设计模式常用于状态变更通知、消息广播等场景。下面通过一个简单的例子展示如何用Golang实现事件的订阅与通知机制。

定义观察者接口与主题

观察者模式包含两个核心部分:一是被观察的主题(Subject),它维护一组观察者并负责通知它们;二是观察者(Observer),它实现更新方法以响应状态变化。

我们先定义接口:

// Observer 观察者接口type Observer interface {    Update(message string)}// Subject 主题接口type Subject interface {    Subscribe(observer Observer)    Unsubscribe(observer Observer)    Notify(message string)}

Observer 接口只有一个 Update 方法,用于接收通知。Subject 接口提供订阅、取消订阅和广播通知的功能。

立即学习“go语言免费学习笔记(深入)”;

实现事件中心

接下来实现一个事件中心,作为具体的主题,管理所有订阅者。

import "sync"// EventCenter 具体的主题实现type EventCenter struct {    observers []Observer    mutex     sync.Mutex}// NewEventCenter 创建新的事件中心func NewEventCenter() *EventCenter {    return &EventCenter{        observers: make([]Observer, 0),    }}// Subscribe 添加观察者func (ec *EventCenter) Subscribe(observer Observer) {    ec.mutex.Lock()    defer ec.mutex.Unlock()    ec.observers = append(ec.observers, observer)}// Unsubscribe 移除观察者func (ec *EventCenter) Unsubscribe(observer Observer) {    ec.mutex.Lock()    defer ec.mutex.Unlock()    for i, obs := range ec.observers {        if obs == observer {            ec.observers = append(ec.observers[:i], ec.observers[i+1:]...)            break        }    }}// Notify 向所有观察者发送通知func (ec *EventCenter) Notify(message string) {    ec.mutex.Lock()    defer ec.mutex.Unlock()    for _, observer := range ec.observers {        observer.Update(message)    }}

使用 sync.Mutex 保证并发安全,避免多个goroutine同时修改观察者列表。

创建具体观察者

定义几个具体的观察者类型,比如邮件通知和日志记录。

// EmailNotifier 邮件通知观察者type EmailNotifier struct{}func (e *EmailNotifier) Update(message string) {    println("发送邮件通知: " + message)}// LogWatcher 日志监听观察者type LogWatcher struct{}func (l *LogWatcher) Update(message string) {    println("记录日志: " + message)}

使用示例

现在将所有部分组合起来,测试事件订阅与通知流程。

func main() {    // 创建事件中心    eventCenter := NewEventCenter()    // 创建观察者    emailNotifier := &EmailNotifier{}    logWatcher := &LogWatcher{}    // 订阅事件    eventCenter.Subscribe(emailNotifier)    eventCenter.Subscribe(logWatcher)    // 发布通知    eventCenter.Notify("系统配置已更新")    // 取消某个订阅    eventCenter.Unsubscribe(emailNotifier)    // 再次通知,只有日志观察者收到    eventCenter.Notify("用户登录成功")}

输出结果:

发送邮件通知: 系统配置已更新记录日志: 系统配置已更新记录日志: 用户登录成功

可以看到,第一次通知时两个观察者都收到了消息;取消 emailNotifier 的订阅后,它不再接收后续通知。

基本上就这些。这个实现简单、清晰,适用于大多数本地事件通知场景。可根据需要扩展支持按事件类型分类、异步通知、带上下文的事件等。不复杂但容易忽略并发安全和取消订阅的边界处理。

以上就是Golang观察者模式事件订阅与通知示例的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/1404607.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年12月15日 20:38:35
下一篇 2025年12月15日 20:38:40

相关推荐

  • Golang基准测试性能优化方法解析

    答案是:Golang性能优化需通过基准测试和pprof分析定位瓶颈,减少内存分配、优化算法、降低锁竞争、提升I/O效率。 Golang的性能优化,说白了,并不是靠感觉或者经验盲目地去改代码,而是一个基于数据、讲究策略的科学过程。它要求我们先通过严谨的基准测试(Benchmark)来找出程序的慢点,再…

    好文分享 2025年12月15日
    000
  • 编写一个Golang程序来计算一个大型文本文件中单词出现的频率

    答案:使用Go语言编写程序,通过bufio逐行读取大文件,结合正则提取单词并用map统计频率,最后按频次降序输出前20个单词。 要编写一个 Go 程序来高效计算大型文本文件中单词的出现频率,可以使用 bufio.Scanner 逐行读取文件,避免将整个文件加载到内存中。同时,使用正则表达式提取单词,…

    2025年12月15日
    000
  • Golang日志收集与云原生监控实践

    使用logrus或zap输出结构化日志,通过Fluent Bit采集至Loki或ES,配合Prometheus监控指标,Grafana统一展示,实现日志、指标、链路三者联动的可观测体系。 在云原生环境下,Golang服务的可观测性至关重要,而日志收集与监控是其中的核心环节。一套高效的日志采集与监控体…

    2025年12月15日
    000
  • Golang使用io/ioutil处理文件内容

    从Go 1.16起,io/ioutil被弃用,其功能移至io和os包。使用ioutil.ReadFile可读取小文件内容到字节切片,适合一次性加载;ioutil.WriteFile支持将数据写入文件并设置权限如0644;ioutil.TempFile用于创建临时文件,需配合defer关闭和删除。尽管…

    2025年12月15日
    000
  • Golang集成测试工具推荐与配置

    答案是Golang集成测试需通过testcontainers-go管理外部依赖以实现环境隔离。它利用Go内置testing包和TestMain进行全局初始化,结合testify断言库提升代码可读性,通过Docker动态启动数据库等服务,确保每次测试在干净、独立环境中运行;同时采用事务回滚、独立Sch…

    2025年12月15日
    000
  • 使用 Go 语言操作 Google Drive SDK v2

    本文将介绍如何使用 Go 语言操作 Google Drive SDK v2。由于官方尚未正式发布 v2 版本的 Go 客户端,我们将指导您如何手动构建客户端,并提供使用示例。通过本文,您将能够使用 Go 语言与 Google Drive v2 API 进行交互,实现文件列表、上传、下载等功能。 手动…

    2025年12月15日
    000
  • Golang使用NSQ实现消息队列处理方法

    答案:在Golang中配置NSQ生产者需引入github.com/nsqio/go-nsq包,创建nsq.Producer实例并连接到nsqd地址如127.0.0.1:4150,使用Publish同步或PublishAsync异步发布消息至指定topic,最后调用Stop优雅关闭。消费者则通过New…

    2025年12月15日
    000
  • admin密码初始密码是多少

    答案:admin初始密码无统一标准,常见为admin、password、123456或空,具体取决于设备类型。最可靠方法是查阅官方文档或查看设备标签,登录后应立即修改密码以避免安全风险。默认密码易遭字典攻击,导致隐私泄露、设备被控等后果。设置强密码推荐使用长密码短语、首字母规则或密码管理器,并为不同…

    2025年12月15日
    000
  • Golang在容器中性能优化与调优方法

    Go程序在容器化部署中需调优以匹配资源限制,首先应设置GOMAXPROCS等于容器CPU限额或启用GOMAXPROCS=0自动适配,避免调度开销;其次控制内存使用,合理配置GOGC以平衡GC频率与内存占用,结合pprof分析内存分配;再者通过精简镜像(如使用Alpine、多阶段构建)减少启动开销;最…

    2025年12月15日
    000
  • 使用 Go 语言操作 Google Drive API v2

    本文档介绍了如何使用 Go 语言操作 Google Drive API v2。由于官方尚未正式发布 v2 版本的 Go 客户端,本文将指导你如何手动构建客户端,并提供基本的使用方法,帮助你快速上手 Google Drive API v2 的开发。 手动构建 Google Drive API v2 G…

    2025年12月15日
    000
  • Golang archive/zip库ZIP文件压缩与解压实践

    Golang的archive/zip库通过zip.Writer和zip.Reader实现ZIP文件的压缩与解压,压缩时需为每个文件创建zip.FileHeader并写入相对路径以保留目录结构,解压时需校验路径防止Zip Slip漏洞;处理大文件时依赖流式I/O避免内存溢出,但需注意磁盘I/O、CPU…

    2025年12月15日
    000
  • Clojure与Java中的Goroutine等价实现:core.async详解

    ! 将数据放入channel(阻塞直到成功)。async/ Select(选择) core.async还提供了一个select!宏,它允许你同时监听多个channel,并在其中任何一个channel准备好时执行相应的操作。这类似于Go的select语句。 示例代码: (require ‘[cloju…

    2025年12月15日
    000
  • Golang网络编程中的数据序列化方法

    JSON适用于跨语言、易读场景,但性能较低;2. Gob为Go专用高效格式,适合内部通信;3. ProtoBuf性能优越,适合大规模分布式系统;应根据通信需求、性能和兼容性权衡选择。 在Golang网络编程中,数据序列化是实现进程间通信的关键环节。由于网络传输只能传递字节流,结构化的数据必须先转换成…

    2025年12月15日
    000
  • Golang使用Gin框架快速搭建Web服务

    Golang搭配Gin框架可快速构建高性能Web服务。首先通过go mod init初始化项目并安装Gin,编写main.go文件创建路由引擎,定义GET和POST接口,使用gin.Default()内置中间件处理日志与异常恢复,通过c.JSON返回响应。生产环境中,可替换默认日志为zap等结构化日…

    2025年12月15日
    000
  • Golang Windows系统下环境变量设置详细步骤

    答案:配置Golang环境变量需设置GOROOT、GOPATH和Path。首先安装Go至默认路径C:Go,此为GOROOT;接着在系统变量中新建GOROOT为C:Go,GOPATH为项目路径如C:UsersYourUserNamego;最后将%GOROOT%bin加入Path。完成后重启命令行,执行…

    2025年12月15日
    000
  • Golang测试中表单输入验证实例

    答案:通过构建用户注册接口并编写测试用例,验证表单输入的合法性,确保用户名、邮箱和年龄符合预设规则,并利用httptest模拟请求检测响应结果是否符合预期。 在Golang中,测试表单输入验证,说白了,就是确保用户提交的数据在到达业务逻辑前,就已经被我们预设的规则筛查过一遍。它主要通过模拟各种合法与…

    2025年12月15日
    000
  • Golang并发日志写入安全实现方法

    答案:Golang中实现并发安全日志写入的核心是避免多个Goroutine同时写文件导致竞态条件,主要方案包括使用sync.Mutex加锁、通过channel异步写入、结合缓冲批量写入,或采用zap等内置并发安全的第三方库。Mutex方案简单但可能成为性能瓶颈;channel方案符合Go并发哲学,能…

    2025年12月15日
    000
  • Golang使用WebSocket库实现实时通信

    答案:Golang通过goroutine和channel实现高效WebSocket通信,利用gorilla/websocket库处理连接升级与消息收发,通过Hub模式集中管理并发连接与广播。 在Golang中实现WebSocket实时通信,核心在于利用其强大的并发模型和成熟的网络库。这通常涉及到升级…

    2025年12月15日
    000
  • Golang Web开发中如何从请求的URL中获取查询参数

    答案:在Golang中通过r.URL.Query()获取URL查询参数,返回url.Values类型,可用Get(“key”)获取单个值,通过query[“key”]获取多值,Go 1.19+支持Has检查存在性。 在Golang的Web开发中,从请求…

    2025年12月15日
    000
  • 理解Golang的happens-before关系在并发同步中的作用

    happens-before关系是Go并发编程的核心,它通过同步原语如goroutine启动、channel通信、互斥锁、sync.WaitGroup、sync.Once和原子操作建立内存操作的可见性顺序,确保共享数据的正确访问。若无明确的happens-before关系,将导致数据竞争和不可预测行…

    2025年12月15日
    000

发表回复

登录后才能评论
关注微信