技术笔记项目:深入理解观察者模式(Observer Pattern)
2025-06-06 18:50:23作者:胡易黎Nicole
什么是观察者模式?
观察者模式是一种行为设计模式,它定义了一种一对多的依赖关系,让多个观察者对象同时监听某一个主题对象。当主题对象状态发生变化时,它会自动通知所有观察者对象,使它们能够自动更新自己。
想象一下报纸订阅的场景:报社(主题)负责发布新闻,而订阅者(观察者)则会在新闻发布时自动收到更新。观察者模式正是这种机制的软件实现。
观察者模式的核心组件
1. 主题(Subject)接口
定义三个核心操作:
attach():添加观察者detach():移除观察者notifyObservers():通知所有观察者
2. 具体主题(ConcreteSubject)
维护观察者列表和状态数据,当状态改变时调用通知方法。
3. 观察者(Observer)接口
定义update()方法,供主题在状态变化时调用。
4. 具体观察者(ConcreteObserver)
实现update()方法,定义收到通知后的具体行为。
观察者模式的两种实现方式
推模型(Push Model)
主题主动将变化的数据推送给观察者:
void update(int temperature, int humidity);
拉模型(Pull Model)
主题仅通知观察者状态已改变,观察者自行获取所需数据:
void update(Subject subject);
推模型更直接但不够灵活,拉模型更灵活但需要观察者主动获取数据。实际开发中可根据需求选择或混合使用。
多语言实现示例
Java实现
// 主题接口
interface Subject {
void attach(Observer o);
void detach(Observer o);
void notifyObservers();
}
// 具体主题:气象站
class WeatherStation implements Subject {
private List<Observer> observers = new ArrayList<>();
private float temperature;
public void setTemperature(float temp) {
this.temperature = temp;
notifyObservers();
}
// 实现接口方法...
}
// 观察者接口
interface Observer {
void update(float temperature);
}
// 具体观察者:显示设备
class DisplayDevice implements Observer {
public void update(float temp) {
System.out.println("温度更新:" + temp);
}
}
JavaScript实现
class NewsPublisher {
constructor() {
this.subscribers = [];
}
subscribe(subscriber) {
this.subscribers.push(subscriber);
}
publish(news) {
this.subscribers.forEach(sub => sub.update(news));
}
}
class Subscriber {
update(news) {
console.log(`收到新闻:${news}`);
}
}
Python实现
class StockMarket:
def __init__(self):
self._investors = []
self._stock_price = None
def attach(self, investor):
self._investors.append(investor)
def price_change(self, new_price):
self._stock_price = new_price
self._notify()
def _notify(self):
for investor in self._investors:
investor.update(self._stock_price)
实际应用场景
- GUI事件处理:按钮点击、键盘输入等事件监听
- 发布-订阅系统:消息队列、新闻推送
- 状态监控:服务器资源监控、应用程序性能监控
- 数据同步:数据库变更通知、缓存更新
- 游戏开发:成就系统、事件触发
观察者模式的优缺点
优点
- 松耦合:主题和观察者之间依赖最小化
- 动态关系:运行时可以随时添加或移除观察者
- 广播通信:一次通知可以触发多个对象的更新
- 符合开闭原则:新增观察者无需修改主题代码
缺点
- 通知顺序不可控:观察者接收通知的顺序不确定
- 性能问题:大量观察者或频繁通知会导致性能下降
- 内存泄漏风险:观察者未正确注销可能导致内存无法释放
- 调试困难:复杂的通知链可能使程序流程难以追踪
与其他模式的关系
- 中介者模式:观察者实现对象间的分布式通信,而中介者集中管理通信
- 责任链模式:观察者模式中所有观察者都会收到通知,而责任链模式中请求可能被某个处理者拦截
- 发布-订阅模式:观察者模式的变体,通常引入消息代理进一步解耦
最佳实践建议
- 考虑线程安全:在多线程环境中使用观察者模式时,确保通知过程是线程安全的
- 避免循环依赖:观察者不应直接或间接依赖其观察的主题
- 控制通知粒度:不要过度使用细粒度通知,考虑批量更新
- 提供取消订阅机制:确保观察者能够正确注销,防止内存泄漏
- 考虑性能优化:对于高频更新场景,可以考虑使用事件队列或节流机制
总结
观察者模式是构建松耦合系统的强大工具,特别适合需要实现对象间动态通知机制的场景。通过本技术笔记项目的详细解析,我们深入理解了观察者模式的核心概念、实现方式以及实际应用中的各种考量。掌握这一模式将帮助你设计出更加灵活、可维护的软件系统。
登录后查看全文
热门项目推荐
atomcodeClaude Code 的开源替代方案。连接任意大模型,编辑代码,运行命令,自动验证 — 全自动执行。用 Rust 构建,极致性能。 | An open-source alternative to Claude Code. Connect any LLM, edit code, run commands, and verify changes — autonomously. Built in Rust for speed. Get StartedRust0117- DDeepSeek-V4-ProDeepSeek-V4-Pro(总参数 1.6 万亿,激活 49B)面向复杂推理和高级编程任务,在代码竞赛、数学推理、Agent 工作流等场景表现优异,性能接近国际前沿闭源模型。Python00
MiMo-V2.5-ProMiMo-V2.5-Pro作为旗舰模型,擅⻓处理复杂Agent任务,单次任务可完成近千次⼯具调⽤与⼗余轮上 下⽂压缩。Python00
GLM-5.1GLM-5.1是智谱迄今最智能的旗舰模型,也是目前全球最强的开源模型。GLM-5.1大大提高了代码能力,在完成长程任务方面提升尤为显著。和此前分钟级交互的模型不同,它能够在一次任务中独立、持续工作超过8小时,期间自主规划、执行、自我进化,最终交付完整的工程级成果。Jinja00
SenseNova-U1-8B-MoT-SFTenseNova U1 是一系列全新的原生多模态模型,它在单一架构内实现了多模态理解、推理与生成的统一。 这标志着多模态AI领域的根本性范式转变:从模态集成迈向真正的模态统一。SenseNova U1模型不再依赖适配器进行模态间转换,而是以原生方式在语言和视觉之间进行思考与行动。Python00
MiniMax-M2.7MiniMax-M2.7 是我们首个深度参与自身进化过程的模型。M2.7 具备构建复杂智能体应用框架的能力,能够借助智能体团队、复杂技能以及动态工具搜索,完成高度精细的生产力任务。Python00
项目优选
收起
暂无描述
Dockerfile
718
4.58 K
deepin linux kernel
C
28
16
Claude Code 的开源替代方案。连接任意大模型,编辑代码,运行命令,自动验证 — 全自动执行。用 Rust 构建,极致性能。 | An open-source alternative to Claude Code. Connect any LLM, edit code, run commands, and verify changes — autonomously. Built in Rust for speed.
Get Started
Rust
769
117
Ascend Extension for PyTorch
Python
584
719
🎉 (RuoYi)官方仓库 基于SpringBoot,Spring Security,JWT,Vue3 & Vite、Element Plus 的前后端分离权限管理系统
Vue
1.63 K
957
本项目是CANN提供的数学类基础计算算子库,实现网络在NPU上加速计算。
C++
975
960
暂无简介
Dart
957
238
openEuler内核是openEuler操作系统的核心,既是系统性能与稳定性的基石,也是连接处理器、设备与服务的桥梁。
C
419
364
AI 将任意文档转换为精美可编辑的 PPTX 演示文稿 — 无需设计基础 | 包含 15 个案例、229 页内容
Python
94
7
本仓将收集和展示高质量的仓颉示例代码,欢迎大家投稿,让全世界看到您的妙趣设计,也让更多人通过您的编码理解和喜爱仓颉语言。
C
442
4.51 K