新闻中心

c++如何实现观察者模式_C++发布订阅机制的设计实例

2025-11-19
浏览次数:
返回列表
观察者模式通过定义一对多依赖关系,实现对象间松耦合通信。1. 使用抽象接口:定义Observer基类和Subject类,通过继承与虚函数实现通知机制,适合需要多态的场景。2. 使用std::function:以函数对象替代继承,支持lambda、函数指针等可调用对象,提升灵活性。3. 智能指针版本:结合shared_ptr与weak_ptr自动管理订阅生命周期,避免悬空指针。适用于GUI事件、MVC架构、消息总线等场景,建议根据需求选择继承或回调方式,并注意线程安全与资源管理。

c++如何实现观察者模式_c++发布订阅机制的设计实例

观察者模式(Observer Pattern),也叫发布-订阅模式,是一种行为设计模式,用于在对象之间定义一对多的依赖关系,当一个对象的状态发生变化时,所有依赖它的对象都会自动收到通知。在 C++ 中实现这一机制,可以通过抽象基类、函数指针、std::function 或智能指针等现代 C++ 特性来完成。

1. 使用抽象接口实现观察者模式

这是最经典的方式,通过定义观察者和被观察者的抽象接口,实现松耦合通信。

步骤说明:

  • 定义一个 Observer 抽象类,包含纯虚函数 update()。
  • 定义一个 Subject 类,维护观察者列表,提供注册、移除和通知接口。
  • 具体观察者继承 Observer,实现自己的 update 行为。

代码示例:

#include <iostream>
#include <vector>
#include <algorithm>
<p>class Observer;</p><p>class Subject {
private:
std::vector<Observer*> observers;
int state;</p><p>public:
void attach(Observer<em> obs);
void detach(Observer</em> obs);
void notify();
void setState(int s);
int getState() const { return state; }
};</p><p>class Observer {
protected:
Subject<em> subject;
public:
explicit Observer(Subject</em> s) : subject(s) {
subject->attach(this);
}
virtual ~Observer() {
subject->detach(this);
}
virtual void update() = 0;
};</p><p>void Subject::attach(Observer* obs) {
observers.push_back(obs);
}</p><p>void Subject::detach(Observer* obs) {
observers.erase(
std::remove(observers.begin(), observers.end(), obs),
observers.end()
);
}</p><p>void Subject::notify() {
for (auto obs : observers) {
obs->update();
}
}</p><p>void Subject::setState(int s) {
state = s;
notify();
}</p><p>// 具体观察者
class ConcreteObserverA : public Observer {
public:
explicit ConcreteObserverA(Subject* s) : Observer(s) {}
void update() override {
std::cout << "Observer A: Subject state is now " << subject->getState() << "\n";
}
};</p><p>class ConcreteObserverB : public Observer {
public:
explicit ConcreteObserverB(Subject* s) : Observer(s) {}
void update() override {
std::cout << "Observer B: Subject state changed to " << subject->getState() << "\n";
}
};</p>

使用方式:

int main() {
    Subject subject;
    ConcreteObserverA observerA(&subject);
    ConcreteObserverB observerB(&subject);
<pre class='brush:php;toolbar:false;'>subject.setState(10);  // 两个观察者都会收到通知
subject.setState(20);

return 0;

}

2. 使用 std::function 实现更灵活的订阅机制

如果不想使用继承,可以用函数对象代替抽象类,让订阅者更加灵活,支持 lambda、函数指针或任意可调用对象。

#include <iostream>
#include <vector>
#include <functional>
#include <algorithm>
<p>class EventSystem {
private:
std::vector<std::function<void(int)>> listeners;</p><p>public:
// 订阅事件
void subscribe(std::function<void(int)> callback) {
listeners.push_back(callback);
}</p><pre class='brush:php;toolbar:false;'>// 发布事件
void publish(int value) {
    for (const auto& listener : listeners) {
        listener(value);
    }
}

};

使用示例:

小云雀 小云雀

剪映出品的AI视频和图片创作助手

小云雀 1949 查看详情 小云雀
int main() {
    EventSystem eventBus;
<pre class='brush:php;toolbar:false;'>// 使用 lambda 订阅
eventBus.subscribe([](int val) {
    std::cout << "Logger: Value updated to " << val << "\n";
});

eventBus.subscribe([](int val) {
    if (val > 50) {
        std::cout << "Alert: High value detected: " << val << "\n";
    }
});

// 模拟数据变化
eventBus.publish(30);
eventBus.publish(60);

return 0;

}

3. 支持自动解绑的智能指针版本(进阶)

为了避免手动管理生命周期导致的悬空指针问题,可以结合 std::shared_ptrweak_ptr 实现自动清理。

思路:每个订阅者返回一个句柄(token),当句柄销毁时自动从列表中移除。或者使用 weak_ptr 存储回调,在调用前检查是否有效。

#include <iostream>
#include <vector>
#include <functional>
#include <memory>
<p>class Publisher {
private:
struct CallbackWrapper {
std::weak_ptr<std::function<void(int)>> cb;
};
std::vector<CallbackWrapper> callbacks;</p><p>public:
std::shared_ptr<std::function<void(int)>> 
subscribe(std::function<void(int)> func) {
auto shared_func = std::make_shared<std::function<void(int)>>(func);
callbacks.push_back({shared_func});
return shared_func;  // 返回共享指针作为“订阅句柄”
}</p><pre class='brush:php;toolbar:false;'>void publish(int value) {
    callbacks.erase(
        std::remove_if(callbacks.begin(), callbacks.end(),
            [value](const CallbackWrapper& wrapper) {
                auto locked = wrapper.cb.lock();
                if (locked) {
                    (*locked)(value);
                    return false;
                }
                return true;  // 已失效,移除
            }),
        callbacks.end()
    );
}

};

这样当外部持有的 shared_ptr 被释放,下次 publish 时会自动清理无效项。

应用场景与建议

观察者模式适用于以下场景:

  • GUI 事件系统(按钮点击通知多个组件)
  • 模型-视图架构中,模型更新通知视图刷新
  • 日志系统、消息总线、状态监控等

设计建议:

- 若需要多态行为,使用接口 + 继承方式。 - 若注重灵活性和轻量级,推荐 std::function + 回调。 - 注意循环引用和生命周期管理,尤其是在使用智能指针时。 - 线程安全需额外加锁(如 std::mutex),若跨线程发布。

基本上就这些。根据项目复杂度选择合适实现方式即可。

以上就是c++++如何实现观察者模式_C++发布订阅机制的设计实例的详细内容,更多请关注其它相关文章!


# app  # ai  # go  # 顺德网站建设制作开发  # 衡阳咨询营销型网站优化  # 浙江百度营销推广公司  # 天津网站营销推广运营  # 不限量关键词seo  # 服务企业建设网站  # 贵阳论坛营销推广资源  # 益阳seo关键词  # 地调局预算 网站建设  # 软文推广炒作营销  # 管理机制  # 何为  # 多态  # 设计实例  # 适用于  # 移除  # 回调  # 都是  # 句柄  # 如何实现  # red  # stream  # ios  # c++ 


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


相关推荐: 一加Ace 6T支持全新明眸护眼:通过了最严苛的护眼小金标认证  正确连接J*aScript到HTML实现可点击图片与自定义事件处理  离线运行Go语言之旅:本地部署与GOPATH配置指南  Adobe PDF表单中利用J*aScript解析与格式化日期组件的教程  网易大神账号申诉需要多久_网易大神账号申诉流程说明  自定义Bag-of-Words实现:处理带负号的词汇权重  俄罗斯方块最新版入口 俄罗斯方块在线玩官网入口  《铁拳8》黑皮辣妹新实机:元气满满的18岁少女!  MAC的“快捷指令”怎么同步到iPhone_MAC利用iCloud同步所有设备的自动化指令  Safari浏览器输入栏卡顿如何解决 Safari搜索建议与缓存清理  Angular中单选按钮的正确使用与常见陷阱解析  12306选座怎么选到特殊座位_12306特殊座位选择注意事项  必由学官网首页入口 必由学教师网页版登录指南  Log4j Console Appender性能瓶颈与高并发优化策略  抖音网页版企业服务中心登录入口_抖音网页版企业登录平台  J*a应用集成GitHub CLI与API认证指南  谷歌浏览器怎么给标签页静音_Chrome标签静音快捷操作  理解J*aScript Promise的微任务队列与执行顺序  电脑屏幕颜色不舒服怎么办_Windows夜间模式与色彩校准教程【护眼技巧】  QQ邮箱官网登录入口 QQ邮箱网页版邮箱快速登录  腾讯视频怎么使用多账号家庭管理_腾讯视频家庭多账号统一管理与权限分配教程  机构:以往存储涨价周期小米利润率实际上有所改善 能转嫁给消费者等  Composer如何处理Git子模块(submodule)依赖_Composer与Git Submodule的对比与选择  J*a应用程序首次运行自动创建文件与目录的最佳实践  Python中高效访问嵌套字典与列表中的键值对  期待已久:小米17 Ultra、小米首款NAS本月登场  搜狗浏览器如何使用密码生成器创建强密码 搜狗浏览器内置密码安全工具  押井守高度称赞《辐射4》:玩了八年都停不下来!  Windows10怎么开启存储感知 Windows10系统设置自动清理临时文件释放C盘空间【教程】  c++项目目录结构应该如何组织_c++工程化项目结构规范  抖音创作助手登录入口_抖音创作辅助工具官网直达  Lar*el 8 多关键词数据库搜索优化实践  AO3网页版最新入口合集 Archive of Our Own在线访问指南  Win11怎么开启高性能模式_Windows 11电源计划优化设置  C#中解析不规范的HTML为XML 常见的坑与解决办法  J*aScript设计模式实践_j*ascript代码优化  Yandex免登录官网入口_俄罗斯Yandex搜索引擎直达链接  C++ explicit关键字防止隐式转换_C++构造函数安全规范  漫蛙Manwa2官网入口地址分享 漫蛙漫画PC版永久访问通道  C++的std::forward_list怎么用_C++ STL中单向链表容器的特点与应用  拼多多视频播放卡顿如何处理 拼多多视频播放优化技巧  怎样更改Windows系统的默认安装路径_避免C盘爆满的终极设置【技巧】  Windows10怎么开启夜间模式 Windows10系统设置调整色温与亮度缓解夜间用眼疲劳【教程】  构建轻量级网站内部消息系统:Formspree 集成指南  Sublime怎么配置Nim语言环境_Sublime Nim代码高亮与补全  Safari自带网页翻译功能怎么用 无需插件轻松看懂外文网站【方法】  c++如何使用Meson构建系统_c++比CMake更快的构建工具  NRF24L01数据传输深度解析:解决大载荷接收异常与分包策略  j*a toString()的覆盖  Go RPC HTTP服务正确实现与常见陷阱解析 

搜索