新闻中心

C++如何实现观察者设计模式_C++中实现对象间发布-订阅的经典模式

2025-11-23
浏览次数:
返回列表
观察者模式通过定义一对多依赖实现对象间松耦合通信,当被观察者状态变化时自动通知所有观察者。示例中TemperatureSensor作为Subject维护Observer列表并触发notify,Display和Logger作为具体观察者重写update响应温度变化,主函数注册监听并模拟数据更新,验证了事件广播机制的正确性与动态解绑功能。

c++如何实现观察者设计模式_c++中实现对象间发布-订阅的经典模式

观察者模式是一种行为设计模式,用于在对象之间定义一对多的依赖关系,当一个对象的状态发生改变时,所有依赖于它的对象都会自动收到通知。这种模式广泛应用于事件处理系统、GUI组件、消息订阅机制等场景。在C++中,可以通过抽象基类与指针(或智能指针)结合的方式优雅地实现发布-订阅机制。

定义观察者与被观察者接口

为了实现松耦合,首先需要定义两个核心角色:观察者(Observer)和被观察者(Subject)。观察者通过继承统一接口,实现更新方法;被观察者维护观察者列表,并在状态变化时通知它们。

示例代码:

Observer.h

class Subject;
<p>class Observer {
public:
virtual ~Observer() = default;
virtual void update() = 0;  // 当被观察者状态改变时调用
};</p>

Subject.h

#include <vector>
#include <algorithm>
#include "Observer.h"
<p>class Subject {
private:
std::vector<Observer*> observers;</p><p>public:
void attach(Observer* obs) {
observers.push_back(obs);
}</p><pre class='brush:php;toolbar:false;'>void detach(Observer* obs) {
    observers.erase(
        std::remove(observers.begin(), observers.end(), obs),
        observers.end()
    );
}

void notify() {
    for (auto* obs : observers) {
        obs->update();
    }
}

};

实现具体观察者与被观察者

接下来创建具体的被观察者类(如TemperatureSensor)和多个具体观察者(如Display、Logger),它们根据通知执行各自逻辑。

示例代码:

TemperatureSensor.h

#include "Subject.h"
<p>class TemperatureSensor : public Subject {
private:
double temperature;</p><p>public:
void setTemperature(double temp) {
temperature = temp;
notify();  // 状态改变,通知所有观察者
}</p><pre class='brush:php;toolbar:false;'>double getTemperature() const {
    return temperature;
}

};

Display.h

PictoGraphic PictoGraphic

AI驱动的矢量插图库和插图生成平台

PictoGraphic 133 查看详情 PictoGraphic
#include "Observer.h"
#include "TemperatureSensor.h"
#include <iostream>
<p>class Display : public Observer {
private:
TemperatureSensor* sensor;</p><p>public:
explicit Display(TemperatureSensor* s) : sensor(s) {}</p><pre class='brush:php;toolbar:false;'>void update() override {
    std::cout << "Display: Current temperature is " 
              << sensor->getTemperature() << "°C\n";
}

};

Logger.h

#include "Observer.h"
#include "TemperatureSensor.h"
#include <iostream>
<p>class Logger : public Observer {
private:
TemperatureSensor* sensor;</p><p>public:
explicit Logger(TemperatureSensor* s) : sensor(s) {}</p><pre class='brush:php;toolbar:false;'>void update() override {
    std::cout << "Log: Temperature updated to " 
              << sensor->getTemperature() << "°C at " 
              << /* 模拟时间戳 */ "12:00:00" << "\n";
}

};

使用示例与注意事项

在主程序中注册观察者,模拟数据变化,验证通知机制是否正常工作。

main.cpp

#include "TemperatureSensor.h"
#include "Display.h"
#include "Logger.h"
<p>int main() {
TemperatureSensor sensor;
Display display(&sensor);
Logger logger(&sensor);</p><pre class='brush:php;toolbar:false;'>sensor.attach(&display);
sensor.attach(&logger);

sensor.setTemperature(25.5);  // 触发通知
sensor.setTemperature(27.0);  // 再次通知

sensor.detach(&display);      // 移除某个观察者
sensor.setTemperature(29.5);  // 只有Logger收到通知

return 0;

}

输出结果:

Display: Current temperature is 25.5°C
Log: Temperature updated to 25.5°C at 12:00:00
Display: Current temperature is 27.0°C
Log: Temperature updated to 27.0°C at 12:00:00
Log: Temperature updated to 29.5°C at 12:00:00

注意点:

  • 被观察者持有观察者的原始指针,需确保观察者生命周期长于被观察者,避免悬空指针。
  • 可改用 std::weak_ptr 配合 std::shared_ptr 管理生命周期,更安全。
  • notify() 过程中若允许动态增删观察者,应考虑线程安全或使用副本遍历。
  • update() 接口可根据需求扩展参数,例如传递具体变化的数据,而非强制查询。

基本上就这些。这个模式在C++中虽无语言级支持,但通过面向对象设计能清晰表达发布-订阅逻辑,适合中小型系统解耦。

以上就是C++如何实现观察者设计模式_C++中实现对象间发布-订阅的经典模式的详细内容,更多请关注其它相关文章!


# 是一种  # 网站建设新闻文案怎么写  # 安达医院网站建设  # 广州网站制作公司优化  # 张家界市网站seo优化排名  # 微推广营销方案案例分享  # 烧饼营销推广方案  # 南宁推广网站推荐  # 热点数据类网站排名优化  # 抖音如何创建营销推广  # 曲靖网站推广外包  # 主程序  # 多个  # go  # 复用  # 多路  # 管理机制  # 何为  # 面向对象  # 都是  # 如何实现  # red  # stream  # ios  # c++  # ai 


相关栏目: 【 科技资讯46185 】 【 网络学院92790


相关推荐: sublime怎么进行远程开发编辑_配置rsub/rmate实现sublime编辑服务器文件  马斯克:Optimus 人形机器人复数形式为 Optimi  sublime如何处理大型CSV文件的列对齐_sublime高级表格编辑插件指南  单射、满射与双射的关系 一文理清所有逻辑  C++ vector二维数组定义_C++ vector of vector用法  J*a递归快速排序中静态变量导致数据累积问题的解决方案  windows10怎么查看硬盘序列号_windows10硬盘id查询命令  快手极速版在线观看 官方网页版登录地址  C++ string find函数返回值npos详解_C++字符串查找失败的判断条件  谷歌浏览器无痕模式怎么开 Chrome开启无痕浏览设置方法【教程】  正确连接J*aScript到HTML实现可点击图片与自定义事件处理  mysql通配符支持数字匹配吗_mysql通配符能否用于数字匹配的解析  python3时间如何用calendar输出?  163邮箱注册官网 免费申请163个人邮箱  在WordPress中通过REST API获取BasicAuth保护的远程文章  C++如何生成随机数_C++ random库使用方法与范围设置  快手官方唯一登录入口 谨防山寨钓鱼网站  Golang如何使用net/url解析URL_Golang URL解析与处理方法  蛙漫2日版入口 WAMAN2(日版)无删减漫画官网链接  Golang如何实现Web文件静态资源服务器_Golang静态资源服务器开发与实践  微博网页版首页入口 微博电脑端官网登录链接  解决Python logging 中 datefmt 导致时间戳固定不变的问题  J*aScript教程:根据元素文本内容动态设置背景色  微信网页版官方快速登录入口 微信网页版网页版账号直达  知乎APP怎么管理已购盐选内容_知乎APP盐选内容购买记录与查看方法  QQ邮箱登录平台入口 QQ邮箱网页版邮箱官方入口  Win11 USB传输速度慢怎么解决 Win11 USB驱动更新与设置  qq游戏免费畅玩入口_qq游戏电脑版快速启动  谷歌浏览器如何快速清除某个网站的数据_Chrome网站缓存清理方法  提升屏幕阅读器对“m”时间单位的播报准确性:HTML与CSS组合解决方案  QQ邮箱在线登录平台 QQ邮箱个人邮箱网页版入口  J*aScript中针对特定容器内图片动画的实现教程  荒野行动PC版怎么注册_荒野行动PC版账号注册详细流程图文教程  哔哩哔哩忘记密码了怎么找回_哔哩哔哩密码找回方法  魅族17怎样用浏览器译外语网页_iPhone魅族17浏览器译外语网页【即时翻译】  铁路12306的积分有效期是多久_铁路12306积分有效期说明  yandex入口引擎手机版 yandex安卓版下载入口  夸克浏览器网页版最新地址 夸克浏览器官方入口合集  AO3同人作品网入口 AO3搜索引擎官网永久地址  163邮箱网页版入口导航平台 163邮箱网页版登录入口官网导航  在J*a中如何使用BigDecimal进行高精度计算_BigDecimal类应用指南  可靠CSGO开箱平台解析 CSGO开箱网合集  汽水音乐在线版入口_汽水音乐网页播放手册  MAC怎么安装Homebrew包管理器_MAC为开发者和高级用户安装命令行工具  C#使用XPath查询节点时出错? 常见语法错误与调试技巧  优化MinIO list_objects_v2 操作的性能瓶颈与最佳实践  抖音网页版企业服务中心登录入口_抖音网页版企业登录平台  漫蛙2正版漫画站 漫蛙2网页版快速访问入口  夸克浏览器图书入口 夸克手机浏览器阅读入口  qq游戏网页版直接玩_qq游戏免下载快速入口 

搜索