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

从C++编程入手设计模式——观察者模式

从C++编程入手设计模式——观察者模式

​ 观察者模式简直就是字如其名,观察观察,观察到了告诉别人。观察手的作用如此,观察者模式的工作机制也是如此。这个模式的核心思路是:一个对象的状态发生变化时,自动通知依赖它的其他对象,让它们自行更新

主要的组成部分

一是被观察者,也叫主题(Subject),它负责记录观察者并在自身状态发生变化时通知它们;换而言之,Subject作为信息的生成端,生成信息。

二是观察者(Observer),它主动注册到被观察者中,等待被通知,一旦收到消息就立即采取行动。这里生成端会使用远程的方式通知我们的被观察者投送信息。

​ 你可以把观察者模式想象成“订阅报纸”。报社就是主题,它维护一个订阅者名单(观察者列表)。每当有新一期的报纸(数据更新)出版时,它就向所有订阅者投递(通知)。每位订阅者收到报纸后根据自己的需要决定怎么处理,是直接阅读、剪报存档还是转发朋友。这种发布-订阅机制就是观察者模式的最基本形式。


在 C++ 中怎么写?

观察者模式的代码实现非常自然。我们会先定义一个抽象的观察者接口(通常叫 Observer 或者 Sender),让不同的具体观察者实现它;再定义一个主题类(通常叫 SubjectNotifier),它包含一个观察者列表,并负责注册、移除和通知观察者。每当主题内部数据变化(比如温度、湿度更新),它就会遍历观察者列表,挨个调用它们的“更新方法”。观察者模式的重点不在于通知的“数据内容”本身,而在于自动通知的机制


选择观察者模式?

在简单的程序中,我们可能会直接调用某些对象来执行逻辑。但如果程序复杂起来,比如某个数据更新后要引发十几种操作,硬编码这些调用就会让系统结构僵硬,维护困难。

观察者模式将“变化的主体”和“依赖的反应者”解耦。被观察者不用关心观察者是谁,它只管发出通知。观察者自己决定是否接收通知、如何响应。这让系统的扩展性和灵活性大大提高

比如天气系统中新加一个“语音播报设备”,只要实现观察者接口并注册进去,其他部分代码几乎无需改动。


观察者模式的几个关键词

  1. 低耦合:主题和观察者之间不互相依赖,只通过接口通信。
  2. 自动同步:当主题数据变化时,所有观察者自动得到通知。
  3. 一对多:一个主题可以对应多个观察者。
  4. 注册/注销机制:观察者可以自由加入或离开通知流程。

Example

​ 可以直接看一个代码框架来解决问题

#include <iostream>
#include <vector>
#include <memory>
#include <string>
#include <algorithm>// 观察者接口
class Observer {
public:virtual void update(const std::string& message) = 0;virtual ~Observer() = default;
};// 具体观察者
class ConcreteObserver : public Observer {
private:std::string name;
public:explicit ConcreteObserver(std::string n) : name(std::move(n)) {}void update(const std::string& message) override {std::cout << "Observer [" << name << "] received: " << message << '\n';}
};// 主题接口
class Subject {
public:virtual void attach(std::shared_ptr<Observer> observer) = 0;virtual void detach(std::shared_ptr<Observer> observer) = 0;virtual void notify(const std::string& message) = 0;virtual ~Subject() = default;
};// 具体主题
class ConcreteSubject : public Subject {
private:std::vector<std::shared_ptr<Observer>> observers;
public:void attach(std::shared_ptr<Observer> observer) override {observers.push_back(std::move(observer));}void detach(std::shared_ptr<Observer> observer) override {observers.erase(std::remove(observers.begin(), observers.end(), observer), observers.end());}void notify(const std::string& message) override {for (const auto& observer : observers) {observer->update(message);}}
};int main() {auto subject = std::make_shared<ConcreteSubject>();auto obs1 = std::make_shared<ConcreteObserver>("Observer1");auto obs2 = std::make_shared<ConcreteObserver>("Observer2");subject->attach(obs1);subject->attach(obs2);subject->notify("Event 1 occurred");subject->detach(obs1);subject->notify("Event 2 occurred");return 0;
}

​ 你可以使用这样的方式来作为事件驱动框架的一个根基。

练习题

天气站系统

要求:设计一个天气数据发布系统,天气站(WeatherStation)是“主题”,可通知多个“观察者”显示设备(如手机、网页等)。

功能要求

  1. 观察者可以注册/注销。
  2. 当温度或湿度变化时,所有注册观察者会收到更新并打印数据。(注:这个部分笔者的参考代码没有实现,您自己酌情思考)

提示

  • 使用接口 IObserver 表示观察者,定义如 update(float temperature, float humidity)
  • 主题接口 ISubject 需要支持 registerObserver()removeObserver()notifyObservers()
  • 可以实现两个观察者类 PhoneDisplayWebDisplay

modern-cpp-patterns-playground/Observer/WeatherForecast at main · Charliechen114514/modern-cpp-patterns-playground

相关文章:

  • TensorFlow 安装与 GPU 驱动兼容(h800)
  • 人工智能学习45-Incep网络
  • 经济法-4- 合同法律制度
  • 从0开始学linux韦东山教程Linux驱动入门实验班(1)
  • Web攻防-XSS跨站Cookie盗取数据包提交网络钓鱼BEEF项目XSS平台危害利用
  • 【软考高级系统架构论文】论软件系统架构风格
  • 【simulink】IEEE5节点系统潮流仿真模型(2机5节点全功能基础模型)
  • 【Java】对象
  • 操作系统内核态和用户态--1-基础认识
  • 操作系统内核态和用户态--2-系统调用是什么?
  • 分布式锁 不同的拒绝策略 应用场景 业务上的思考
  • QT vscode cmake 编译 undefined reference to `vtable for 问题解决
  • 自定义 Hook:在 Vue3 中复用逻辑
  • 【C++】pybind11:生成 Python 可用的动态库
  • 5.3 VSCode使用FFmpeg库
  • 国家级与省级(不含港澳台)标准地图服务网站汇总
  • python的校园兼职系统
  • 修改SSH端口实战
  • 【51单片机2位数码管100毫秒的9.9秒表】2022-5-16
  • man的使用
  • 视频网站模板下载/成都企业seo
  • 网站建设上市公司/推广团队在哪里找
  • 湘西网站建设吧/长沙专业竞价优化公司
  • 不是用于制作网页的软件/seo关键字优化
  • 企业画册设计图片/成都关键词优化报价
  • 做网站的属于什么行业/搜索引擎优化排名关键字广告