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

【设计模式】Head First 设计模式——观察者模式 C++实现

设计模式最大的作用就是在变化和稳定中间寻找隔离点,然后分离它们,从而管理变化。将变化像小兔子一样关到笼子里,让它在笼子里随便跳,而不至于跳出来把你整个房间给污染掉。

设计思想

主题对象(出版者)管理某些数据,当主题内的数据改变,就会通知观察者(订阅者)。

观察者模式定义了对象之间的一对多依赖,当一个对象改变状态时,它的所有依赖者都会收到通知并自动更新。相比让多个对象控制同一份数据,可以得到更干净的OO设计。

业务场景

假定现在有一个气象站类,这个气象站类会不定时更新温度,湿度,气压的最新数据,要求你在这些数值变化时,立即通知一些气象局进行展示,而他们得到的数据是一样的,但是采用的展示手段可能是不一样的,比如有的采用图示,有的采用文字,有的采用语音播报等,给出设计代码。

代码案例

#include <iostream>
#include <vector>
#include <algorithm>using namespace std;// 观察者抽象类
class Observer
{
public:virtual void update(const float tmp, const float hum, const float pre) = 0;virtual ~Observer() {}
};// 展示抽象类
class Display
{
public:virtual void display() const = 0;virtual ~Display() {}
};// 主题抽象类
class Subject
{
public:virtual void registerObserver(Observer* b) = 0;virtual void removeObserver(Observer* b)   = 0;virtual void notifyObserver() const        = 0;virtual ~Subject() {}
};class WeatherData : public Subject
{
private:std::vector<Observer*> observers;float                  temperature = 0;float                  humidity    = 0;float                  pressure    = 0;public:WeatherData() : Subject() {}void registerObserver(Observer* b){observers.push_back(b);}void removeObserver(Observer* ob){auto it = find(observers.begin(), observers.end(), ob);if (it != observers.end()){observers.erase(it);}}// 通知观察者,使得所有观察者更新信息。void notifyObserver() const{for (auto& it : observers){it->update(temperature, humidity, pressure);}}void measurementsChanged(){notifyObserver();}// 设置气象站天气信息并通知观察者。void setMeasurements(const float tmp, const float hum, const float pre){temperature = tmp;humidity    = hum;pressure    = pre;measurementsChanged();}~WeatherData(){for (auto& each : observers){delete each;each = nullptr;}}
};class CurrentConditionsDisplay : public Observer, public Display
{
private:float    temperature = 0;float    humidity    = 0;Subject* weatherData = nullptr;public:// 初始化信息板,并在气象站对其登记。CurrentConditionsDisplay(Subject* wd) : Observer(), Display(){weatherData = wd;weatherData->registerObserver(this);}// 更新天气信息并展示。void update(const float tmp, const float hum, const float pre){temperature = tmp;humidity    = hum;display();}void display() const{cout << "气象局1: 当前温度:" << temperature << ", 当前湿度:" << humidity << endl;}~CurrentConditionsDisplay(){if (weatherData){delete weatherData;weatherData = nullptr;}}
};class CurrentConditionsDisplay2 : public Observer, public Display
{
private:float    temperature = 0;float    humidity    = 0;Subject* weatherData = nullptr;public:// 初始化信息板,并在气象站对其登记。CurrentConditionsDisplay2(Subject* wd) : Observer(), Display(){weatherData = wd;weatherData->registerObserver(this);}// 更新天气信息并展示。void update(const float tmp, const float hum, const float pre){temperature = tmp;humidity    = hum;display();}void display() const{cout << "气象局2: 当前温度:" << temperature << ", 当前湿度:" << humidity << endl;}~CurrentConditionsDisplay2(){if (weatherData){delete weatherData;weatherData = nullptr;}}
};int main()
{WeatherData*               wd  = new WeatherData();CurrentConditionsDisplay*  cd  = new CurrentConditionsDisplay(wd);CurrentConditionsDisplay2* cd2 = new CurrentConditionsDisplay2(wd);// wd修改天气信息 并通知展示板wd->setMeasurements(80, 65, 30.4f);wd->setMeasurements(60, 35, 32.1f);wd->removeObserver(cd);cout << "~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~" << endl;wd->setMeasurements(60, 35, 31.1f);return 0;
}
http://www.lryc.cn/news/144593.html

相关文章:

  • 【ES】笔记-Promise基本使用
  • 服务器数据恢复-reiserfs文件系统损坏如何恢复数据?
  • 直播预告:把脉2023年下半场—主动防御邮箱盗号威胁
  • 专题:平面、空间直线参数方程下的切线斜率问题
  • JavaScript—对象与构造方法
  • 微信小程序社区户口管理的系统设计与实现
  • 闲人闲谈PS之四十六——网络生产全流程
  • 如何在VR头显端实现低延迟的RTSP或RTMP播放
  • 【工具类】提高办公效率(兼具有趣、好玩)
  • navicat连接数据库的方法(易懂)
  • 收支明细管理实操:如何准确记录并修改收支明细?
  • SSL证书的工作原理是怎样的?
  • Java发送请求到第三方(RestTemplate方法)
  • CentOS 7 Nacos 设置开机自动重启
  • 安防监控平台EasyCVR视频汇聚平台增加首页告警类型的详细介绍
  • 构建安全可信、稳定可靠的RISC-V安全体系
  • 3.RabbitMQ 架构以及 通信方式
  • 分布式事务篇-2.1 阿里云轻量服务器--Docker--部署Seata
  • C语言这么没用??
  • Docker运维篇
  • 【数学建模】清风数模正课7 多元线性回归模型
  • 文心一言 VS 讯飞星火 VS chatgpt (83)-- 算法导论8.1 4题
  • 温故知新之:代理模式,静态代理和动态代理(JDK动态代理)
  • 软件工程(十二) 设计模式之创建型模式
  • 使用docker、docker-compose部署微服务
  • 【Axure高保真原型】中继器网格图片拖动摆放
  • 《基于 Vue 组件库 的 Webpack5 配置》4. 压缩 CSS 和 js 文件
  • electron globalShortcut 快捷键,在焦点移到其他软件上时,调用快捷键报错
  • 【PHP】PHP条件控制
  • 超干货!Linux中断响应流程