当前位置: 首页 > news >正文

【设计模式】观察者

观察者模式

1 简介

观察者模式是观察者对象们通过注册到被观察者对象中,从而使被观察者发生变化时能通知到观察者,避免硬编码,使用写死的代码逻辑调用通知,从而实现解耦效果。

2 基本代码逻辑

观察者

class IObserver {
public:virtual ~IObserver() = 0;virtual void update(const Type &type) = 0;
};

被观察者(Subject)

class ISubject {
public:virtual ~ISubject() = 0;virtual void attach(IObserver *observer) = 0;virtual void detach(IObserver *observer) = 0;virtual void notify(const Type &type) = 0;
};

具体实现

class ConcreteObserver : public IObserver {
public:void update(const Type &type) override {// ...}
};
class ConcreteSubject : public ISubject {
private:vector<IObserver*> observers_{};public:void attach(IObserver *observer) override {observers_.emplace_back(observer);}void detach(IObserver *observer) override {observers_.erase(std::remove(observers_.begin(), observers_.end(), observer), observers_.end());}void notify(const Type &type) override {for (auto *observer : observers_) {observer->update(type);}}void setAttr(const Type &type) {// ... procnotify(type);}
};

3 拓展:支持多事件类型的分发

当被观察者(Subject)需要处理 多种不同类型的事件,且每种事件需要独立的观察者列表时,可以通过 事件类型区分泛型接口 实现更灵活的设计。
以下是针对多事件场景的优化方案:

设计思路

  1. 事件类型标识
    使用枚举或字符串定义不同的事件类型(如 EventType::WeatherEventType::Temperature)。
  2. 分事件管理观察者
    在基类中用 std::mapstd::unordered_map 维护不同事件对应的观察者列表。
  3. 泛型事件数据传递
    通过模板或基类 EventData 封装不同事件的数据(如温度值、天气描述)。
  4. 类型安全的观察者接口
    为不同事件类型定义专门的观察者接口,或通过 dynamic_cast 实现运行时类型检查。
1. 定义事件类型与数据基类
#include <iostream>
#include <vector>
#include <unordered_map>
#include <string>
#include <memory>// 事件类型枚举
enum class EventType {WeatherUpdate,   // 天气更新TemperatureChange, // 温度变化// 可扩展更多事件类型...
};// 事件数据基类(泛化数据传递)
class EventData {
public:virtual ~EventData() = default;
};
2. 具体事件数据类(按需扩展)
// 天气事件数据
class WeatherData : public EventData {
public:std::string description;explicit WeatherData(const std::string& desc) : description(desc) {}
};// 温度事件数据
class TemperatureData : public EventData {
public:double value;explicit TemperatureData(double val) : value(val) {}
};
3. 观察者接口(支持多事件类型)
class IObserver {
public:virtual ~IObserver() = default;// 统一处理事件的方法(通过类型区分)virtual void onEvent(EventType type, const EventData& data) = 0;
};
4. 被观察者基类(分事件管理观察者)
class Subject {
private:// 按事件类型存储观察者列表std::unordered_map<EventType, std::vector<IObserver*>> observers_;public:virtual ~Subject() = default;// 注册观察者到特定事件类型void attach(EventType type, IObserver* observer) {observers_[type].push_back(observer);}// 注销观察者void detach(EventType type, IObserver* observer) {auto& list = observers_[type];list.erase(std::remove(list.begin(), list.end(), observer), list.end());}// 通知特定事件类型的观察者void notify(EventType type, const EventData& data) {if (observers_.find(type) == observers_.end()) return;for (auto* observer : observers_[type]) {observer->onEvent(type, data);}}
};

5. 具体被观察者实现(例如气象站)

class WeatherStation : public Subject {
public:// 触发天气更新事件void setWeather(const std::string& weather) {WeatherData data(weather);notify(EventType::WeatherUpdate, data);}// 触发温度变化事件void setTemperature(double temp) {TemperatureData data(temp);notify(EventType::TemperatureChange, data);}
};

6. 具体观察者实现(按需订阅事件)

// 天气显示屏(仅关注天气更新)
class WeatherDisplay : public IObserver {
public:void onEvent(EventType type, const EventData& data) override {if (type != EventType::WeatherUpdate) return; // 过滤无关事件const auto& weatherData = dynamic_cast<const WeatherData&>(data);std::cout << "[天气显示屏] 最新天气: " << weatherData.description << std::endl;}
};// 温度报警器(仅关注温度变化)
class TemperatureAlarm : public IObserver {
public:void onEvent(EventType type, const EventData& data) override {if (type != EventType::TemperatureChange) return;const auto& tempData = dynamic_cast<const TemperatureData&>(data);if (tempData.value > 35.0) {std::cout << "[温度报警] 警告!当前温度: " << tempData.value << "°C" << std::endl;}}
};

7. 客户端使用示例

int main() {WeatherStation station;  // 被观察者WeatherDisplay display;  // 观察者1:订阅天气TemperatureAlarm alarm;  // 观察者2:订阅温度// 注册观察者到特定事件station.attach(EventType::WeatherUpdate, &display);station.attach(EventType::TemperatureChange, &alarm);// 触发事件station.setWeather("晴");      // 通知天气观察者station.setTemperature(28.5);  // 通知温度观察者station.setTemperature(37.0);  // 触发报警return 0;
}

关键优化点

  1. 事件类型区分
    通过 EventType 明确划分不同行为,观察者仅处理订阅的事件。
  2. 泛化数据传递
    EventData 基类允许传递任意派生类数据,结合 dynamic_cast 确保类型安全。
  3. 动态扩展性
    添加新事件只需扩展 EventType 和对应的 EventData 子类,无需修改基类。
  4. 精准通知
    每个事件类型独立维护观察者列表,避免无关观察者被调用。

适用场景

  • 多事件系统:如游戏引擎(角色移动、攻击、死亡等事件独立通知)。
  • 模块化监控:不同模块关注系统的不同状态变化。
  • 可配置订阅:允许观察者动态选择关注的事件类型。

相关文章:

  • RabbitMQ 为什么引入 Exchange 的概念, 交换机有什么作用.
  • QML与C++:基于ListView调用外部模型进行增删改查(性能优化版)
  • 【Leetcode-Hot100】最大子数组和
  • 【教程】如何使用Labelimg查看已经标注好的YOLO数据集标注情况
  • C++| 深入剖析std::list底层实现:链表结构与内存管理机制
  • VTK知识学习(51)- 交互与Widget(三)
  • 小程序获取用户总结(全)
  • ArrayList vs LinkedList,HashMap vs TreeMap:如何选择最适合的集合类?
  • CEPH配置优化建议
  • 小程序css实现容器内 数据滚动 无缝衔接 点击暂停
  • AtomNet:在极端MCU约束下基于算子设计微型模型
  • LivePortrait 使用指南:让静态照片“动”起来的魔法工具
  • 【自动化测试】如何获取cookie,跳过登录的简单操作
  • 一个异步架构设计:批量消费RabbitMQ,批量写入Elasticsearch(golang实现)
  • hadoop执行sqoop任务找不到jar
  • Dijkstra算法求解最短路径—— 从零开始的图论讲解(2)
  • 第十章 go mod操作
  • 【Java SE】Collections类详解
  • 2.1 腾讯校招通关指南-算法与数据结构
  • trl的安装与单GPU多GPU测试
  • 赵作海因病离世,妻子李素兰希望过平静生活
  • 协会:坚决支持司法机关依法打击涉象棋行业的违法行为
  • 第一集|《刑警的日子》很生活,《执法者们》有班味
  • 欧盟公布关税反制清单,瞄准美国飞机、汽车等产品
  • 越秀地产前4个月销售额约411.2亿元,达年度销售目标的34.1%
  • 央行:全力推进一揽子金融政策加快落地