go实现观察者模式核心是接口+切片+方法绑定,需用sync.RWMutex保障并发安全,推荐异步通知并recover单个panic,支持函数式Observer和topic分类的EventBus。

用 Go 实现观察者模式,核心是解耦事件发布者(Subject)与订阅者(Observer),不依赖第三方库,靠接口 + 切片 + 方法绑定就能搞定。重点不是“模拟 java 风格”,而是利用 Go 的简洁性把通知逻辑写得清晰、安全、可扩展。
定义 Observer 接口和 Subject 基础结构
Observer 只需一个 Update 方法,接收任意事件数据;Subject 管理观察者列表和通知逻辑。不需要泛型也能保持类型安全——事件数据用 Interface{} 或自定义事件结构体传入,由具体 Observer 自行断言。
- Observer 接口建议定义为:type Observer interface { Update(event interface{}) }
- Subject 可设计为结构体,含 observers []Observer 和 mu sync.RWMutex(多协程场景必须加锁)
- 添加/移除观察者方法要加写锁,通知时用读锁避免阻塞其他 Subscribe/Unsubscribe 调用
支持异步通知与错误隔离
同步通知容易因某个 Observer 执行慢或 panic 拖垮整个流程。推荐默认走 goroutine 异步通知,并 recover 单个 panic,不让一个失败影响其余观察者。
- Notify 方法内部对每个 observer 启一个 goroutine:go func(obs Observer) { defer func(){…}(); obs.Update(event) }(o)
- 用 sync.WaitGroup 控制批量通知完成(如需等待全部结束)
- 若需结果反馈(比如某些 Observer 必须成功),可额外提供 NotifySync() 同步版本,按需选用
用闭包或函数类型简化简单场景
不是所有情况都需要定义完整结构体。对于一次性监听或脚本类逻辑,直接用函数作为 Observer 更轻量:
立即学习“go语言免费学习笔记(深入)”;
- 定义类型:type EventHandler func(event interface{})
- 实现 Update 方法:func (f EventHandler) Update(e interface{}) { f(e) }
- 注册时写:subject.Subscribe(EventHandler(func(e interface{}) { log.Println(“got:”, e) }))
结合 channel 实现事件总线(进阶可选)
当系统中事件种类多、模块分散,可封装一个全局事件总线(EventBus),底层用 channel + map[String][]Observer 分类管理主题(topic)。发布时按 topic 投递,Observer 只监听关心的主题。
- Subscribe(topic string, obs Observer) 注册到指定 topic
- Publish(topic string, event interface{}) 向该 topic 所有 Observer 广播
- 注意:channel 容量和缓冲策略要合理,避免阻塞;高吞吐下可考虑 ring buffer 或分片 topic
基本上就这些。Go 里观察者模式不复杂但容易忽略并发安全和错误传播,抓住接口抽象、锁粒度、异步隔离三个点,就能写出健壮又易读的事件通知逻辑。