C++如何实现一个观察者设计模式?(代码示例)

发布时间 - 2025-12-30 00:00:00    点击率:
观察者模式在C++中通过抽象基类定义Observer接口并用容器管理观察者,Subject持有一组Observer智能指针,在状态变化时调用其update()实现松耦合;Observer需继承含纯虚update()的基类,Subject用vector维护列表,提供注册、移除和通知功能。

观察者模式(Observer Pattern)在 C++ 中可以通过抽象基类定义接口,用容器管理多个观察者,再通过通知机制实现松耦合的事件响应。核心是让被观察者(Subject)持有一组观察者(Observer)指针,并在状态变化时调用它们的更新方法。

定义观察者接口

所有观察者需继承统一接口,确保被观察者能以多态方式调用其 update() 方法:

class Observer {
public:
    virtual ~Observer() = default;
    virtual void update(const std::string& message) = 0;
};

实现被观察者基类(Subject)

维护观察者列表,提供注册、移除和通知功能。使用 std::vector 存储智能指针(如 std::shared_ptr),避免裸指针生命周期问题:

#include 
#include 
#include 

class Subject {
protected:
    std::vector> observers;

public:
    void attach(std::shared_ptr obs) {
        if (obs) observers.push_back(obs);
    }

    void detach(std::shared_ptr obs) {
        observers.erase(
            std::remove(observers.begin(), observers.end(), obs),
            observers.end()
        );
    }

    void notify(const std::string& message) {
        for (const auto& obs : observers) {
            if (obs) obs->update(message);
        }
    }
};

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

例如一个温度传感器(TemperatureSensor)作为被观察者,两个不同用途的观察者(日志记录器、显示屏)响应变化:

class TemperatureSensor : public Subject {
private:
    double temperature = 0.0;

public:
    void setTemperature(double temp) {
        temperature = temp;
        notify("Temperature changed to " + std::to_string(temp) + "°C");
    }
};

class Logger : public Observer {
public:
    void update(const std::string& message) override {
        std::cout << "[LOG] " << message << std::endl;
    }
};

class Display : public Observer {
public:
    void update(const std::string& message) override {
        std::cout << "[DISPLAY] " << message << std::endl;
    }
};

使用示例

组合对象并触发通知:

int main() {
    TemperatureSensor sensor;
    auto logger = std::make_shared();
    auto display = std::make_shared();

    sensor.attach(logger);
    sensor.attach(display);

    sensor.setTemperature(25.5); // 输出两条消息
    sensor.detach(logger);
    sensor.setTemperature(26.0); // 只有 display 响应

    return 0;
}

注意点:使用 std::shared_ptr 管理观察者生命周期更安全;若观察者可能早于被观察者销毁,可改用 std::weak_ptr 配合检查;C++17 后也可考虑用 std::any 或模板支持泛型通知数据,但基础版本保持简单清晰即可。


# go  # c++  # red  # 多态  # 指针  # 继承  # 接口  # 泛型  # 对象  # 事件  # 传感器  # 移除  # 记录器  # 多个  # 并在  # 也可  # 可以通过  # 两条  # 用它  # 早于 


相关栏目: 【 网站优化151355 】 【 网络推广146373 】 【 网络技术251813 】 【 AI营销90571


相关推荐: Laravel如何实现API速率限制?(Rate Limiting教程)  Laravel如何设置自定义的日志文件名_Laravel根据日期或用户ID生成动态日志【技巧】  Swift中循环语句中的转移语句 break 和 continue  如何将凡科建站内容保存为本地文件?  Laravel如何配置和使用缓存?(Redis代码示例)  网站制作大概多少钱一个,做一个平台网站大概多少钱?  美食网站链接制作教程视频,哪个教做美食的网站比较专业点?  如何确保西部建站助手FTP传输的安全性?  Win11搜索不到蓝牙耳机怎么办 Win11蓝牙驱动更新修复【详解】  javascript中的try catch异常捕获机制用法分析  如何为不同团队 ID 动态生成多个“认领值班”按钮  Laravel用户密码怎么加密_Laravel Hash门面使用教程  详解MySQL数据库的安装与密码配置  做企业网站制作流程,企业网站制作基本流程有哪些?  Laravel如何实现全文搜索功能?(Scout和Algolia示例)  中国移动官方网站首页入口 中国移动官网网页登录  香港服务器网站搭建教程-电商部署、配置优化与安全稳定指南  网站制作怎么样才能赚钱,用自己的电脑做服务器架设网站有什么利弊,能赚钱吗?  如何在阿里云香港服务器快速搭建网站?  JavaScript常见的五种数组去重的方式  Laravel如何集成微信支付SDK_Laravel使用yansongda-pay实现扫码支付【实战】  如何登录建站主机?访问步骤全解析  html5audio标签播放结束怎么触发事件_onended回调方法【教程】  什么是javascript作用域_全局和局部作用域有什么区别?  Laravel怎么自定义错误页面_Laravel修改404和500页面模板  Python函数文档自动校验_规范解析【教程】  Android实现代码画虚线边框背景效果  Laravel的辅助函数有哪些_Laravel常用Helpers函数提高开发效率  JavaScript如何操作视频_媒体API怎么控制播放  学生网站制作软件,一个12岁的学生写小说,应该去什么样的网站?  Laravel项目结构怎么组织_大型Laravel应用的最佳目录结构实践  齐河建站公司:营销型网站建设与SEO优化双核驱动策略  如何快速使用云服务器搭建个人网站?  Bootstrap CSS布局之列表  laravel怎么配置Redis作为缓存驱动_laravel Redis缓存配置教程  香港服务器租用每月最低只需15元?  阿里云网站搭建费用解析:服务器价格与建站成本优化指南  品牌网站制作公司有哪些,买正品品牌一般去哪个网站买?  javascript基于原型链的继承及call和apply函数用法分析  如何在香港免费服务器上快速搭建网站?  Laravel如何自定义错误页面(404, 500)?(代码示例)  深圳网站制作的公司有哪些,dido官方网站?  Laravel如何安装Breeze扩展包_Laravel用户注册登录功能快速实现【流程】  如何快速查询域名建站关键信息?  如何在建站之星绑定自定义域名?  如何制作公司的网站链接,公司想做一个网站,一般需要花多少钱?  如何挑选优质建站一级代理提升网站排名?  网站制作软件有哪些,制图软件有哪些?  个人摄影网站制作流程,摄影爱好者都去什么网站?  购物网站制作费用多少,开办网上购物网站,需要办理哪些手续?