观察者模式-委托(大话设计模式)C/C++版本

news2025/1/10 20:58:57

观察者模式-委托

先看该常规的没有委托概念的代码,如下:

非委托

#include <iostream>
#include <string>
#include <list>
using namespace std;

class Subject; // 前向声明

// 抽象观察者
class Observer
{
protected:
    string name;
    Subject *sub;

public:
    Observer(string name, Subject *sub)
    {
        this->name = name;
        this->sub = sub;
    }
    virtual void update() = 0;
};

// 抽象通知者
class Subject
{
protected:
    list<Observer *> observers;

public:
    string event;
    virtual void attach(Observer *) = 0;
    virtual void detach(Observer *) = 0;
    virtual void notify() = 0;
};

// 具体通知者,秘书
class Secretary : public Subject
{
    void attach(Observer *observer) override
    {
        observers.push_back(observer);
    }

    void detach(Observer *observer) override
    {
        list<Observer *>::iterator iter = observers.begin();
        while (iter != observers.end())
        {
            if ((*iter) == observer)
            {
                observers.erase(iter);
            }
            ++iter;
        }
    }

    void notify() override
    {
        list<Observer *>::iterator iter = observers.begin();
        while (iter != observers.end())
        {
            (*iter)->update();
            ++iter;
        }
    }
};

// 具体的观察者,看股票的
class StockObserver : public Observer
{
public:
    StockObserver(string name, Subject *sub) : Observer(name, sub)
    {
    }
    void update() override
    {
        cout << name << " 收到消息:" << sub->event << endl;
        if (sub->event == "梁所长来了!")
        {
            cout << "我马上关闭股票,装做很认真工作的样子!" << endl;
        }
    }
};

// 具体的观察者,看NBA的
class NBAObserver : public Observer
{
public:
    NBAObserver(string name, Subject *sub) : Observer(name, sub)
    {
    }
    void update() override
    {
        cout << name << " 收到消息:" << sub->event << endl;
        if (sub->event == "梁所长来了!")
        {
            cout << "我马上关闭NBA,装做很认真工作的样子!" << endl;
        }
    }
};

int main()
{
    // 创建通知者
    Subject *dwq = new Secretary();

    // 被观察的对象
    Observer *xs = new NBAObserver("xiaoshuai", dwq);
    Observer *lm = new StockObserver("limin", dwq);

    // 加入观察队列
    dwq->attach(xs);
    dwq->attach(lm);

    // 事件
    dwq->event = "去吃饭了!";
    // 通知
    dwq->notify();
    cout << endl;

    // 事件
    dwq->event = "梁所长来了!";
    // 通知
    dwq->notify();
    cout << endl;

    return 0;
}

观察者为什么要和委托有联系?

  1. 抽象通知类还是依赖抽象观察者类
  2. 不同的被观察者可能并不相关,所以设计的接口不一定全都是一样的接口名称update.
    总结这两点,就是去除抽象观察者这个基类后,并且该更新状态的接口不同名的情况下,观察者模式仍旧可以成立。

何谓委托(大概了解即可)

委托是一种设计原则,指的是一个对象(委托者)将自身的一部分职责或功能交由另一个对象(被委托者)来执行。这种机制允许对象专注于自己的核心功能,同时利用外部对象的专业能力来完成辅助任务。委托强调的是行为的传递和执行,而不是对象所有权或生命周期的管理
在这里插入图片描述

那么,C++如何实现委托?

function+bind
参考:https://blog.csdn.net/qq_38410730/article/details/103637778
https://www.cnblogs.com/linuxAndMcu/p/14576162.html#_label1

# include <functional>
std::function<函数类型>

std::function是一个可调用对象包装器,是一个类模板,可以容纳除了类成员函数指针之外的所有可调用对象,它可以用统一的方式处理函数、函数对象、函数指针,并允许保存和延迟它们的执行。

# include <functional>
std::bind(函数指针, 绑定对象);

std::bind可以看作一个通用的函数适配器,它接受一个可调用对象,生成一个新的可调用对象来适应原对象的参数列表

这样就可以实现:直接将具体观察者实例和观察者类状态更新接口函数(即之前的update)绑定。通知类管理的通知对象就从观察者实例变成了观察者的函数对象了。

也就是达到了:通知类不再依赖抽象观察者类的update接口,而是直接依赖这个绑定的函数对象。这样就可以达到:不需要抽象观察者类并且可以使用不同的接口名。

委托

参考:https://blog.csdn.net/weixin_43272766/article/details/94166777

  1. 去除了抽象观察者类
  2. 使用了C++的function+bind实现委托
#include <iostream>
#include <string>
#include <functional>
#include <list>
using namespace std;

// 
typedef function<void()> Action;

// 抽象观察者 无

// 抽象通知者
class Subject
{
protected:
    list<Action *> actions;

public:
    string event;
    virtual void attach(Action *a) = 0;
    virtual void detach(Action *a) = 0;
    virtual void notify() = 0;
};

// 具体通知者,秘书
class Secretary : public Subject
{
public:
    void attach(Action *a) override
    {
        actions.push_back(a);
    }

    void detach(Action *a) override
    {
        actions.remove(a);
    }

    void notify() override
    {
        for (auto act : actions)
        {
            (*act)();
        }
    }
};

// 具体的观察者,看股票的
class StockObserver
{
private:
    string name;
    Subject *sub;

public:
    StockObserver(string _name, Subject *_sub) : name(_name), sub(_sub)
    {
    }

    void StockObserverUpdate()
    {
        cout << name << " 收到消息:" << sub->event << endl;
        if (sub->event == "梁所长来了!")
        {
            cout << "我马上关闭股票,装做很认真工作的样子!" << endl;
        }
    }
};

// 具体的观察者,看NBA的
class NBAObserver
{
private:
    string name;
    Subject *sub;

public:
    NBAObserver(string _name, Subject *_sub) : name(_name), sub(_sub)
    {
    }

    void NBAObserverUpdate()
    {
        cout << name << " 收到消息:" << sub->event << endl;
        if (sub->event == "梁所长来了!")
        {
            cout << "我马上关闭股票,装做很认真工作的样子!" << endl;
        }
    }
};

int main()
{
    // 创建通知者
    Subject *dwq = new Secretary();

    // 被观察的对象
    NBAObserver *xs = new NBAObserver("xiaoshuai", dwq);
    Action acA = bind(&NBAObserver::NBAObserverUpdate, xs);
    StockObserver *lm = new StockObserver("limin", dwq);
    Action acB = bind(&StockObserver::StockObserverUpdate, lm);

    // 加入观察者队列
    dwq->attach(&acA);
    dwq->attach(&acB);

    // 事件
    dwq->event = "去吃饭了!";
    // 通知
    dwq->notify();
    cout << endl;

    // 事件
    dwq->event = "梁所长来了!";
    // 通知
    dwq->notify();
    cout << endl;

    return 0;
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1835415.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

上证50etf期权的手续费要多少钱?期权懂分享

今天带你了解上证50etf期权的手续费要多少钱&#xff1f;上证50ETF期权的开户一般交易手续费默认为7元一张。在进行期权开户之前&#xff0c;建议提前联系一名券商的客户经理&#xff0c;协商期权手续费优惠事宜。 上证50etf期权的手续费要多少钱&#xff1f; 上证50ETF期权的…

Spring是如何设计IOC容器的?BeanFactory ApplicationContext

BeanFactory是Spring框架中最底层的接口&#xff0c;用于实例化、配置和管理bean。它使用控制反转&#xff08;IOC&#xff09;模式&#xff0c;将对象的创建、管理和装配的职责从应用程序代码中转移给Spring容器。这样&#xff0c;应用程序代码就无需关心对象如何创建和装配&a…

一篇文章了解常用排序算法

排序 文章目录 排序直接(插入)排序InsertSort思想实现方法&#xff1a; 希尔排序ShellSort&#xff08;可过OJ)思想预排序gap的作用整体代码 选择排序SelectSort思想完整代码 堆排序HeapSort(可过OJ)思想大根堆向下调整 完整代码 冒泡排序BubbleSort快速排序&#xff08;快排&a…

Autodesk Inventor 机械三维设计软件下载安装,Inventor 专业的三维制图软件

Inventor&#xff0c;它的一大亮点在于能够将三维尺寸、标注以及尺寸公差直接融入三维模型中&#xff0c;使得这些关键信息能够无缝对接下游应用&#xff0c;极大地提升了设计流程中的连贯性和一致性。 谈及Inventor的尺寸公差功能&#xff0c;更是让人赞不绝口。在复杂的设计过…

测试:设计测试用例

文章目录 概念设计正交法判定表法 本篇总结的是测试用例的概念和设计方法 概念 测试用例是为了实施测试而向被测试的系统提供的一组集合&#xff0c;这个集合中包含的内容有测试环境&#xff0c;操作步骤&#xff0c;测试数据&#xff0c;预期结果等要素 在测试用例的设计中…

第 54 期:MySQL Too many open files 报错

社区王牌专栏《一问一实验&#xff1a;AI 版》全新改版归来&#xff0c;得到了新老读者们的关注。其中不乏对 ChatDBA 感兴趣的读者前来咨询&#xff0c;表达了想试用体验 ChatDBA 的意愿&#xff0c;对此我们表示感谢 &#x1f91f;。 目前&#xff0c;ChatDBA 还在最后的准备…

【差分数组】2772. 使数组中的所有元素都等于零

本文涉及知识点 差分数组 LeetCode2772. 使数组中的所有元素都等于零 给你一个下标从 0 开始的整数数组 nums 和一个正整数 k 。 你可以对数组执行下述操作 任意次 &#xff1a; 从数组中选出长度为 k 的 任一 子数组&#xff0c;并将子数组中每个元素都 减去 1 。 如果你可…

[dataworks]从mysql导入数据

一、新建离线同步 在ods的数据集成下点新建-->离线同步 1、起名imp_t_ods_uc_cst_terminal_dtl_df 前缀imp是import的缩写 t代表trade即MySQL的交易库(trade)的简写 ods即导入到ods层 uc_cst_terminal_dt为MySQL对应的表名 df为日全量导入&#xff08;di为日增量导入&…

Profibus协议转Modbus协议网关模块帮助PLC实现智能激光设备通讯

一、前言 Profibus转Modbus网关&#xff08;XD-MDPB100&#xff09;是一种工业通信协议转换设备&#xff0c;用于实现Profibus协议与Modbus协议之间的转换。Profibus转Modbus网关在工业自动化系统中具有广泛的应用&#xff0c;它解决了不同协议设备之间的通信问题。本文将深入…

半监督医学图像分割:基于对抗一致性学习和动态卷积网络的方法| 文献速递-深度学习结合医疗影像疾病诊断与病灶分割

Title 题目 Semi-Supervised Medical Image Segmentation Using Adversarial Consistency Learning and Dynamic Convolution Network 半监督医学图像分割&#xff1a;基于对抗一致性学习和动态卷积网络的方法 01 文献速递介绍 医学图像分割在计算辅助诊断和治疗研究中扮演…

M41T00串行实时时钟-国产兼容RS4C1339

RS4C1340是一种实时时钟&#xff08;RTC&#xff09;/日历&#xff0c;与ST M41T00引脚兼容&#xff0c;功能等效&#xff0c;包括软件时钟校准。该器件还提供VBAT引脚上的涓流充电能力、较低的计时电压和振荡器STOP标志。寄存器映射的块访问与ST设备相同。涓流充电器和标志需要…

HarmonyOS 页面路由(Router)

1. HarmonyOS页面路由(Router) 页面路由指在应用程序中实现不同页面之间的跳转和数据传递。HarmonyOS提供了Router模块&#xff0c;通过不同的url地址&#xff0c;可以方便地进行页面路由&#xff0c;轻松地访问不同的页面。本文将从页面跳转、页面返回和页面返回前增加一个询问…

LeetCode刷题之HOT100之单词拆分

上午把docker基础学完了。下午来了闲的无聊&#xff0c;做一题先。 1、题目描述 2、逻辑分析 这个问题是一个典型的动态规划问题&#xff0c;我们可以使用一个布尔数组 dp 来记录字符串 s 的前缀是否可以被拆分成字典中的单词。具体地&#xff0c;dp[i] 表示字符串 s 的前 i …

Odrivegui 、odrivetool运行时的几个问题(windows)

ODrivetool 遇到的几个问题 错误信息 Traceback (most recent call last): File “c:\Users\hpf\Desktop\import matplotlib.py”, line 1, in import matplotlib.pyplot as plt File “C:\Users\hpf\AppData\Local\Programs\Python\Python39\lib\site-packages\matplotlib_…

【STM32】使用标准库点亮LED

1.硬件设计 LED1的阴极接到了PC13引脚上&#xff0c;我们控制PC13引脚的电平输出状态&#xff0c;即可控制LED1的亮灭。 2.编程要点 使能GPIO端口时钟&#xff1b;初始化GPIO目标引脚为推挽输出模式&#xff1b;编写简单测试程序&#xff0c;控制GPIO引脚输出高、低电平。 查…

数据驱动决策:工单统计工具如何赋能企业精准运营

在当今这个数字化飞速发展的时代&#xff0c;企业对于内部运营效率的追求已经达到了前所未有的高度。你是否曾为了繁杂的工单统计管理而头疼不已&#xff1f;是否曾因为无法准确进行工单统计数据而错失商机&#xff1f;今天&#xff0c;我将向你展示一款革命性的工单统计工具&a…

Python基础教程——20个让人眼前一亮的逻辑妙用!

文末免费赠精品编程资料~~ Python不仅仅是一种编程语言&#xff0c;它还是解决问题的艺术&#xff0c;充满了让人拍案叫绝的“小巧思”。通过这15个小技巧&#xff0c;你不仅能提升编程技能&#xff0c;还能让你的代码更加优雅、高效。让我们一探究竟吧&#xff01; 1. 列表推…

Thinkphp校园新闻发布系统源码 毕业设计项目实例

Thinkphp校园新闻发布系统源码 毕业设计项目实例 校园新闻发布系统模块&#xff1a; 用户模块&#xff1a;注册&#xff0c;登陆&#xff0c;查看个人信息&#xff0c;修改个人信息&#xff0c;站内搜索&#xff0c;新闻浏览等功能&#xff0c; 后台管理员模块&#xff1a;会员…

挖矿宝藏之开发者模式

目录 一、开发者模式简介 二、启动方式 三、元素&#xff08;Elements&#xff09; 四、控制台&#xff08;Console&#xff09; 五、来源&#xff08;Sources&#xff09; 六、网络&#xff08;Network&#xff09; 七、性能&#xff08;Performance&#xff09; 八、…

谷粒商城实战(043集群学习-mysql集群-分库分表)

Java项目《谷粒商城》架构师级Java项目实战&#xff0c;对标阿里P6-P7&#xff0c;全网最强 总时长 104:45:00 共408P 此文章包含第364p-第p365的内容 分库分表 这种基本无人用 shardingSphere shard&#xff08;碎片&#xff09; sphere &#xff08;球&#xff09; sh…