【C++ | 设计模式】观察者模式的详解与实现

2024-09-02 03:36

本文主要是介绍【C++ | 设计模式】观察者模式的详解与实现,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

1.概念

观察者模式(Observer Pattern)是一种行为型设计模式,它的核心思想是定义对象间的一对多依赖关系,当一个对象的状态发生变化时,所有依赖于它的对象都会收到通知并自动更新。这个模式在现实生活中非常常见,比如新闻订阅、社交媒体的推送通知等。

举个简单的例子想象一下,你订阅了一个YouTube频道。当这个频道发布新视频时,你就会收到通知。这就是观察者模式的一个实际应用。YouTube频道相当于“被观察者”,你作为订阅者则是“观察者”。

2. 模式结构

UML结构图:

6c93ad72c2884daba9f1ec5bfb0cab29.png

  • Subject(被观察者/主题):持有对观察者的引用,可以增加、删除、通知观察者。当它的状态发生变化时,会主动通知所有的观察者。
  • Observer(观察者):定义一个接口,用于接收被观察者的通知并更新自身。当被观察者的状态变化时,观察者会接收到通知并执行相关操作。
  • ConcreteSubject(具体的被观察者):具体的被观察者对象,通常包含一些重要的数据,当数据发生变化时,会通知观察者。
  • ConcreteObserver(具体的观察者):具体的观察者对象,负责实现观察者接口,并在被观察者状态变化时作出响应。

工作流程

  1. 订阅/注册:观察者向被观察者注册自己,表示对其状态变化感兴趣。
  2. 状态改变:当被观察者的状态发生变化时,它会调用 notify() 方法,通知所有已注册的观察者。
  3. 通知/更新:观察者在接收到通知后,调用 update() 方法,从而获取被观察者的最新状态并更新自己的状态。

观察者模式的优点

  • 解耦:观察者和被观察者之间的依赖性很低,它们之间通过接口进行通信,可以轻松地添加、移除观察者,系统的可扩展性强。
  • 灵活性:可以在运行时动态地增加或移除观察者,观察者模式使得观察者的数量、种类都可以变化。
  • 一致性:被观察者可以保证所有观察者的状态和它自身的状态是一致的。

观察者模式的缺点

  • 性能开销:如果观察者数量众多,或者被观察者的状态更新频繁,通知所有观察者可能会带来较大的性能开销。
  • 通知顺序问题:在多个观察者的情况下,通知的顺序可能会影响程序的行为,而观察者模式通常不会对通知顺序做出严格规定。
  • 循环依赖风险:如果被观察者和观察者之间存在复杂的依赖关系,可能会导致循环更新或通知,从而引发问题。

适用场景

  • 状态变化的事件通知:当一个对象的状态变化需要通知其他对象时,例如用户界面的数据绑定。
  • 多级联动更新:多个对象之间需要保持一致性,例如在MVC架构中,模型数据变化需要通知视图更新。
  • 发布-订阅系统:例如消息推送系统,发布者发布消息后,所有订阅者都会收到通知。

现实生活中的例子

  • 新闻订阅:用户订阅某个新闻类别,当有新文章发布时,用户会收到通知。
  • 社交媒体:用户关注某个社交媒体账号,当该账号发布新内容时,用户会收到推送通知。
  • 气象站:气象站采集到新数据后,会通知所有的显示设备更新天气信息。

3.案例代码分析:

案例描述

假设我们有一个天气预报系统,系统有一个气象站(WeatherStation),它可以实时检测温度和湿度。用户可以通过不同的显示设备(如手机App、桌面小部件、电子看板等)查看天气信息。

当气象站检测到天气数据变化时,它会通知所有已注册的显示设备更新显示信息。这就是典型的观察者模式场景:气象站是“被观察者”,而显示设备是“观察者”。

代码实现

下面是使用C++实现的天气预报系统的观察者模式代码,包括详细注释。

#include <iostream>
#include <vector>
#include <algorithm>// 观察者接口:定义更新接口,用于接收通知
class Observer {
public:virtual void update(float temperature, float humidity) = 0;  // 更新函数,接收温度和湿度
};// 被观察者接口:定义注册、移除、通知观察者的方法
class Subject {
public:virtual void registerObserver(Observer* o) = 0;  // 注册观察者virtual void removeObserver(Observer* o) = 0;    // 移除观察者virtual void notifyObservers() = 0;              // 通知所有观察者
};// 具体的被观察者:气象站
class WeatherStation : public Subject {
private:std::vector<Observer*> observers;  // 观察者列表float temperature;                 // 当前温度float humidity;                    // 当前湿度public:// 注册观察者void registerObserver(Observer* o) override {observers.push_back(o);}// 移除观察者void removeObserver(Observer* o) override {observers.erase(std::remove(observers.begin(), observers.end(), o), observers.end());}// 通知所有观察者void notifyObservers() override {for (Observer* observer : observers) {observer->update(temperature, humidity);  // 通知每个观察者更新}}// 模拟天气数据的更新void setMeasurements(float temp, float hum) {temperature = temp;humidity = hum;notifyObservers();  // 数据更新后通知观察者}
};// 具体的观察者1:手机App显示器
class PhoneDisplay : public Observer {
public:void update(float temperature, float humidity) override {std::cout << "Phone Display - Temperature: " << temperature << "°C, Humidity: " << humidity << "%\n";}
};// 具体的观察者2:桌面小部件
class DesktopWidget : public Observer {
public:void update(float temperature, float humidity) override {std::cout << "Desktop Widget - Temperature: " << temperature << "°C, Humidity: " << humidity << "%\n";}
};// 具体的观察者3:电子看板
class ElectronicBillboard : public Observer {
public:void update(float temperature, float humidity) override {std::cout << "Electronic Billboard - Temperature: " << temperature << "°C, Humidity: " << humidity << "%\n";}
};int main() {WeatherStation weatherStation;  // 创建气象站对象PhoneDisplay phoneDisplay;      // 创建观察者对象:手机App显示器DesktopWidget desktopWidget;    // 创建观察者对象:桌面小部件ElectronicBillboard billboard;  // 创建观察者对象:电子看板weatherStation.registerObserver(&phoneDisplay);   // 注册观察者到气象站weatherStation.registerObserver(&desktopWidget);weatherStation.registerObserver(&billboard);// 模拟天气数据的更新weatherStation.setMeasurements(25.0f, 65.0f);weatherStation.setMeasurements(30.5f, 70.0f);// 移除一个观察者,模拟观察者取消订阅的情况weatherStation.removeObserver(&desktopWidget);// 再次更新天气数据,观察剩余观察者的反应weatherStation.setMeasurements(28.0f, 55.0f);return 0;
}

 

 

这篇关于【C++ | 设计模式】观察者模式的详解与实现的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1128882

相关文章

golang版本升级如何实现

《golang版本升级如何实现》:本文主要介绍golang版本升级如何实现问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录golanwww.chinasem.cng版本升级linux上golang版本升级删除golang旧版本安装golang最新版本总结gola

MySQL 中的 CAST 函数详解及常见用法

《MySQL中的CAST函数详解及常见用法》CAST函数是MySQL中用于数据类型转换的重要函数,它允许你将一个值从一种数据类型转换为另一种数据类型,本文给大家介绍MySQL中的CAST... 目录mysql 中的 CAST 函数详解一、基本语法二、支持的数据类型三、常见用法示例1. 字符串转数字2. 数字

SpringBoot中SM2公钥加密、私钥解密的实现示例详解

《SpringBoot中SM2公钥加密、私钥解密的实现示例详解》本文介绍了如何在SpringBoot项目中实现SM2公钥加密和私钥解密的功能,通过使用Hutool库和BouncyCastle依赖,简化... 目录一、前言1、加密信息(示例)2、加密结果(示例)二、实现代码1、yml文件配置2、创建SM2工具

Mysql实现范围分区表(新增、删除、重组、查看)

《Mysql实现范围分区表(新增、删除、重组、查看)》MySQL分区表的四种类型(范围、哈希、列表、键值),主要介绍了范围分区的创建、查询、添加、删除及重组织操作,具有一定的参考价值,感兴趣的可以了解... 目录一、mysql分区表分类二、范围分区(Range Partitioning1、新建分区表:2、分

MySQL 定时新增分区的实现示例

《MySQL定时新增分区的实现示例》本文主要介绍了通过存储过程和定时任务实现MySQL分区的自动创建,解决大数据量下手动维护的繁琐问题,具有一定的参考价值,感兴趣的可以了解一下... mysql创建好分区之后,有时候会需要自动创建分区。比如,一些表数据量非常大,有些数据是热点数据,按照日期分区MululbU

MyBatis-Plus 中 nested() 与 and() 方法详解(最佳实践场景)

《MyBatis-Plus中nested()与and()方法详解(最佳实践场景)》在MyBatis-Plus的条件构造器中,nested()和and()都是用于构建复杂查询条件的关键方法,但... 目录MyBATis-Plus 中nested()与and()方法详解一、核心区别对比二、方法详解1.and()

Spring IoC 容器的使用详解(最新整理)

《SpringIoC容器的使用详解(最新整理)》文章介绍了Spring框架中的应用分层思想与IoC容器原理,通过分层解耦业务逻辑、数据访问等模块,IoC容器利用@Component注解管理Bean... 目录1. 应用分层2. IoC 的介绍3. IoC 容器的使用3.1. bean 的存储3.2. 方法注

MySQL 删除数据详解(最新整理)

《MySQL删除数据详解(最新整理)》:本文主要介绍MySQL删除数据的相关知识,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧... 目录一、前言二、mysql 中的三种删除方式1.DELETE语句✅ 基本语法: 示例:2.TRUNCATE语句✅ 基本语

MySQL中查找重复值的实现

《MySQL中查找重复值的实现》查找重复值是一项常见需求,比如在数据清理、数据分析、数据质量检查等场景下,我们常常需要找出表中某列或多列的重复值,具有一定的参考价值,感兴趣的可以了解一下... 目录技术背景实现步骤方法一:使用GROUP BY和HAVING子句方法二:仅返回重复值方法三:返回完整记录方法四:

Python内置函数之classmethod函数使用详解

《Python内置函数之classmethod函数使用详解》:本文主要介绍Python内置函数之classmethod函数使用方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地... 目录1. 类方法定义与基本语法2. 类方法 vs 实例方法 vs 静态方法3. 核心特性与用法(1编程客