【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

相关文章

C#使用yield关键字实现提升迭代性能与效率

《C#使用yield关键字实现提升迭代性能与效率》yield关键字在C#中简化了数据迭代的方式,实现了按需生成数据,自动维护迭代状态,本文主要来聊聊如何使用yield关键字实现提升迭代性能与效率,感兴... 目录前言传统迭代和yield迭代方式对比yield延迟加载按需获取数据yield break显式示迭

Python实现高效地读写大型文件

《Python实现高效地读写大型文件》Python如何读写的是大型文件,有没有什么方法来提高效率呢,这篇文章就来和大家聊聊如何在Python中高效地读写大型文件,需要的可以了解下... 目录一、逐行读取大型文件二、分块读取大型文件三、使用 mmap 模块进行内存映射文件操作(适用于大文件)四、使用 pand

python实现pdf转word和excel的示例代码

《python实现pdf转word和excel的示例代码》本文主要介绍了python实现pdf转word和excel的示例代码,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价... 目录一、引言二、python编程1,PDF转Word2,PDF转Excel三、前端页面效果展示总结一

Python xmltodict实现简化XML数据处理

《Pythonxmltodict实现简化XML数据处理》Python社区为提供了xmltodict库,它专为简化XML与Python数据结构的转换而设计,本文主要来为大家介绍一下如何使用xmltod... 目录一、引言二、XMLtodict介绍设计理念适用场景三、功能参数与属性1、parse函数2、unpa

mac中资源库在哪? macOS资源库文件夹详解

《mac中资源库在哪?macOS资源库文件夹详解》经常使用Mac电脑的用户会发现,找不到Mac电脑的资源库,我们怎么打开资源库并使用呢?下面我们就来看看macOS资源库文件夹详解... 在 MACOS 系统中,「资源库」文件夹是用来存放操作系统和 App 设置的核心位置。虽然平时我们很少直接跟它打交道,但了

C#实现获得某个枚举的所有名称

《C#实现获得某个枚举的所有名称》这篇文章主要为大家详细介绍了C#如何实现获得某个枚举的所有名称,文中的示例代码讲解详细,具有一定的借鉴价值,有需要的小伙伴可以参考一下... C#中获得某个枚举的所有名称using System;using System.Collections.Generic;usi

Go语言实现将中文转化为拼音功能

《Go语言实现将中文转化为拼音功能》这篇文章主要为大家详细介绍了Go语言中如何实现将中文转化为拼音功能,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 有这么一个需求:新用户入职 创建一系列账号比较麻烦,打算通过接口传入姓名进行初始化。想把姓名转化成拼音。因为有些账号即需要中文也需要英

C# 读写ini文件操作实现

《C#读写ini文件操作实现》本文主要介绍了C#读写ini文件操作实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧... 目录一、INI文件结构二、读取INI文件中的数据在C#应用程序中,常将INI文件作为配置文件,用于存储应用程序的

关于Maven中pom.xml文件配置详解

《关于Maven中pom.xml文件配置详解》pom.xml是Maven项目的核心配置文件,它描述了项目的结构、依赖关系、构建配置等信息,通过合理配置pom.xml,可以提高项目的可维护性和构建效率... 目录1. POM文件的基本结构1.1 项目基本信息2. 项目属性2.1 引用属性3. 项目依赖4. 构

C#实现获取电脑中的端口号和硬件信息

《C#实现获取电脑中的端口号和硬件信息》这篇文章主要为大家详细介绍了C#实现获取电脑中的端口号和硬件信息的相关方法,文中的示例代码讲解详细,有需要的小伙伴可以参考一下... 我们经常在使用一个串口软件的时候,发现软件中的端口号并不是普通的COM1,而是带有硬件信息的。那么如果我们使用C#编写软件时候,如