C++|设计模式(七)|⭐️观察者模式与发布/订阅模式,你分得清楚吗

本文内容来源于B站:
【「观察者模式」与「发布/订阅模式」,你分得清楚吗?】

文章目录

  • 观察者模式(Observer Pattern)的代码优化
  • 观察者模式 与 发布订阅模式 他们是一样的吗?
    • 发布订阅模式
    • 总结

我们想象这样一个场景:
我们需要开发一个软件系统,该系统可以实时接收某支股票的最新价格,系统可以控制,将这个价格展示在路边的电子广告牌上,同时,坐在控制室中的管理人员可以在显示器上看到这个价格。
当每支股票价格发生变化时,电子广告牌和显示器上的股票价格也要实时变化。

我们首先使用C++来模拟他的一种可能的实现方式:

  • 引入必要的头文件;
  • 创建一个 Monitor 类来表示显示器,其中包含 print 方法来更新显示器的显示内容;
  • 创建一个 Billboard 类来表示广告牌,同样已有 print 方法来进行更新;
  • 创建 Stock 类,该类封装了与股票相关的数据和操作,它包含有一个表示当前股票价格的成员变量 price,以及一个用于更改其值的成员方法 setPrice。
  • 既然 Stock 类是通过 setPrice 来更新股票价格的,那么我们也可以在这里调用 Monitor 和 Billboard 类中的接口来同步更新广告牌和显示器上的数据,所以我们在 Stock 类找那个加入这两个类中的指针,来实现接口的调用。
  • 需要补充的是,我们需要在 Stock 类中初始化这里的两个类指针,让他们指向已经创建好的 Monitor 和 Billboard 类对象。
#include <iostream>
struct Monitor {void print(int v) {std::cout << "Monitor:" << v;}
};
struct Billboard {void display(int v) {std::cout << "Billboard:" << v;}
};
struct Stock {int price = 20;Monitor* monitor;Billboard* billboard;Stock(Monitor* monitor, Billboard* billboard): monitor(monitor), billboard(billboard)void setPrice(int v) {price = v;monitor->print(price);billboard->print(price);}
};

随后我们就可以调用 Stock 对象上的 setPrice 方法来更新股票价格,并且广告牌与显示器上的数字也会被相应更新。

int main () {Monitor monitor;Billboard billboard;Stock stock {&monitor, &billboard};stock.setPrice(10);
}

但是这样的实现方式有许多的问题:

  1. 类之间的紧耦合:Stock 类的稳定性依赖于 Monitor 与 Billboard 类接口的稳定,而当这些接口的名称或使用方式发生变化时,那么 Stock 类就需要被同时修改;并且同样的情况也发生在有更多的显示媒介加入时,比如软件希望同时支持在手机APP上显示最新的股票价格,这个时候我们也需要再次修改 Stock 类。
  2. 这样的紧耦合使得代码的维护成本变得很高,那么如何解决呢?答案就是观察者模式!

观察者模式(Observer Pattern)的代码优化

也就是我们的 Monitor 类和 Billboard 类都是观察者,这里我们可以提供一个统一的父类 Observer ,并且在里面写入 update 接口,这个接口将所有观察者在观测时发生的不同动作进行了统一的包装。Observer 的构造和析构会在后面进行补全:

#include <iostream>
struct Observer {Observer();virtual ~Observer();virtual void update(int) = 0;
}
struct Monitor {void print(int v) {std::cout << "Monitor:" << v;}
};
struct Billboard {void display(int v) {std::cout << "Billboard:" << v;}
};

然后我们的 Monitor 类的构造函数也做相应的修改,然后我们在 Monitor 类中实现父类的 update 接口,Billboard 观察者类也同理,所有观察者都需要将当观测数据发生变化时,可能发生的行为封装在统一的 update(int) 接口中。

struct Monitor {Monitor() : Observer() {}void print(int v) {std::cout << "Monitor:" << v;}void update(int v) override {print(v);}
};
struct Billboard {Billboard() : Observer() {}void display(int v) {std::cout << "Billboard:" << v;}void update(int v) override {display(v);}
};

下面我们把 Stock 类也加入到代码中,我们首先为他提供一个前置声明。接着我们在 Observer 类中加入一个指向 Stock 类对象的指针引用,随后我们改写 Monitor 和 Billboard 类的构造函数,接着我们将之前那段 Stock 类的定义原封不动得放入到代码中。

#include <iostream>struct Stock;struct Observer {Stock* stock;Observer(Stock* stock);virtual ~Observer();virtual void update(int) = 0;
};
struct Monitor {Monitor(Stock* stock) : Observer(stock) {}void print(int v) {std::cout << "Monitor:" << v;}
};
struct Billboard {Billboard(Stock* stock) : Observer(stock) {}void display(int v) {std::cout << "Billboard:" << v;}
};struct Stock {int price = 0;void setPrice(int v) {price = v;}
};

在观察者模式下, Stock 类作为所有观察者关注的对象,它需要支持三个成员方法,notify(int) detach(Observer*) attach(Observer*),这三个方法会围绕 Stock 类内部维护的一个集合进行操作。这个集合中就存放有指向 Oberser 子类对象的指针。这些指针指向的就是所有对 Stock 类感兴趣的观察者对象。

#include list
struct Stock {int price = 0;std::list<Observer*> observerList;void attach(Observer* o) {observerList.push_back(0);}void detach(Observer* o) {observerList.remove(0);}void setPrice(int v) {price = v;}
};

这里我们的 notify(int v) 方法是观察者模式的核心,这个方法会遍历观察者指针容器,并以此调用每个观察者对象上的 update 方法。通过这种方式,Stock 类便能够在状态发生变化时及时通知所有对此感兴趣的观察者对象来进行相应的更新操作,最后我们在 setPrice 方法中调用 notify 方法,这样在股价发生变化时, 能够通知到所有观察者对象。

#include list
struct Stock {int price = 0;std::list<Observer*> observerList;void attach(Observer* o) {observerList.push_back(0);}void detach(Observer* o) {observerList.remove(0);}void notify(int v) {for (auto observer : observerList) {observer->update(v);}}void setPrice(int v) {price = v;notify(v);}
};

下一步我们需要补全 Observer 类的构造函数,让每一个观察者对象在完成构造后,都能被加入到某个 Stock 对象的观察者集合中。而当观察者对象被析构时则从相应的观察者集合中被移除。

Observer::observer(Stock* stk) : stock(stk) {stock->attach(this);
}
Observer::observer(Stock* stk) : stock(stk) {stock->detach(this);
}

最后在 main 函数中我们可以这样来使用这些类:

int main () {Stock stock;Monitor monitor { &stock };Billboard board { &stock };stock.setPrice(10);
}

我们可以看到当 Stock 类在构造时,不再依赖任何观察者对象,因此它的实现可以保持稳定。而观察者对象也可以自由选择观察目标,可以是不同的 Stock 对象,同时不同种类的观察者对象的横向扩展也变得更加灵活

UML类图如上,并且在某些情况下我们可以进一步对观察者模式进行抽象。从而得到另一个版本的 UML 类图和另一个版本的代码实现。

我们为 Stock 类提供了独立的抽象接口 Subject ,从而将 attach、detach 以及 notify 这三个标准接口抽离出来。而 Stock 则保有自己的内部状态,为了保证状态访问的合法性,他们仅能够通过专有的访问器进行访问,下面是对应的 C++ 代码实现:

#include <iostream>
#include <list>struct Observer;
struct Subject {std::list<Observer*> observerList;virtual void attach(Observer* o) = 0;virtual void detach(Observer* o) = 0;virtual void notify() = 0;
};class Stock : public Subject {int price = 0;public:int getPrice();void setPrice(int);void attach(Observer* o) override;void detach(Observer* o) override;void notify() override;
};struct Observer {Subject* sub;Observer(Subject* sub);virtual ~Observer();virtual void update() = 0;
};struct Monitor : Observer {Monitor(Subject* sub) : Observer(sub) {}void print(int v) const { std::cout << "Monitor: " << v << std::endl; }void update() override { print(static_cast<Stock*>(sub)->getPrice()); }
};struct Billboard : Observer {Billboard(Stock* stock) : Observer(stock) {}void display(int v) const { std::cout << "Billboard: " << v << std::endl; }void update() override { display(static_cast<Stock*>(sub)->getPrice()); }
};int Stock::getPrice(void) { return price; }
void Stock::setPrice(int v) {price = v;notify();
}
void Stock::attach(Observer* o) { observerList.push_back(o); }
void Stock::detach(Observer* o) { observerList.remove(0); }
void Stock::notify() {for (auto observer : observerList) {observer->update();}
}Observer::Observer(Subject* sub) : sub(sub) { sub->attach(this); }Observer::~Observer() { sub->detach(this); }int main() {Stock stock;Monitor monitor{&stock};Billboard board{&stock};stock.setPrice(10);
}

观察者模式 与 发布订阅模式 他们是一样的吗?

发布订阅模式

发布订阅模式是一种常用的系统架构模式,他用来规定一个系统中的不同部分之间应该如何进行消息传递。

在这个模式中,发布者可以是系统中的消息服务、事件服务;而订阅者可以是系统中的其他服务,比如网关服务、路由服务等等。

发布者与订阅者之间互相不知道对方的存在,他们之间通过名为消息代理的部分连接起来。发布者会向消息代理发送不同类型的消息,比如一个系统通知、一个发生的事件。而订阅者则会与消息代理通信,选择自己想要订阅的消息类型。

在之后的流程中,每当消息代理接收到符合要求的消息,便会把他们直接转发给相应的订阅者进行处理;这便是发布订阅模式的基本形式。

总结

所以综合来看,我们可以得出这样的结论:

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

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

批量下载 B 站 视频的工具 downkyi

批量下载 B 站 视频的工具 downkyi 亲测好用 图片&#xff1a; 下载地址&#xff1a; https://github.com/leiurayer/downkyi

MES系统在机床产业智能化的作用

MES系统&#xff08;Manufacturing Execution System&#xff0c;制造执行系统&#xff09;在机床产业智能化过程中发挥着至关重要的作用。以下是万界星空科技MES系统在机床产业智能化中的几个关键作用&#xff1a; 1. 实时数据采集与分析 数据采集&#xff1a;MES系统通过与生…

jenkins获取sonarqube质量门禁结果

前景 在使用 Jenkins 集成 SonarQube 时&#xff0c;获取质量门禁&#xff08;Quality Gate&#xff09;结果非常重要。SonarQube 的质量门禁是一种质量控制机制&#xff0c;用于评估代码质量是否符合预设的标准。以下是获取质量门禁结果的意义和作用&#xff1a; 评估代码质量…

navicat 17 安装

百度网盘 链接: https://pan.baidu.com/s/1nFFQzWhjxRUM_X6bVlWNGw?pwd8888 提取码: 8888 1.双击运行安装包 2.点击下一步 2.勾选我同意&#xff0c;点击下一步 3.自定义安装路径&#xff0c;点击下一步 4.注意勾选桌面快捷方式&#xff0c;点击下一步 5.点击安装 6.点击完…

[渗透测试学习] Editorial-HackTheBox

文章目录 Editorial-HackTheBox信息搜集漏洞利用权限提升参考文章Editorial-HackTheBox 信息搜集 nmap扫描端口 nmap -sV -sC -v 10.10.11.20扫描结果如下 PORT STATE SERVICE VERSION 22/tcp open ssh OpenSSH 8.9p1 Ubuntu 3ubuntu0.7 (Ubuntu Linux; protocol 2.…

组蛋白乳酸化和RNA甲基化如何联动?请大数据把这个思路推给科研人

在细胞生物学中&#xff0c;基因表达调控是决定细胞功能与命运的核心过程之一。组蛋白作为修饰性蛋白&#xff0c;在调控基因转录中起着至关重要的作用。近年来&#xff0c;科学家们发现&#xff0c;组蛋白的多种化学修饰&#xff08;如甲基化、乙酰化、磷酸化等&#xff09;影…

day8 Excel教程——利用数据验证规范单元格输入!(超多干货)

day8 Excel教程——利用数据验证规范单元格输入! 本章目录 day8 Excel教程——利用数据验证规范单元格输入!1. 整数,小数和文本长度1.1 限制用户输入1(最小)和10(最大)之间的数字1.2 限制用户应输入小于或等于3%的小数(百分比)值1.3 限制用户应输入文本长度为18位的身份…

2024年最全数据库开发必备神器:DataGrip 使用介绍

DataGrip 使用介绍 前言 DataGrip 是由 JetBrains 开发的一款专业的数据库管理工具&#xff0c;广泛支持多种数据库系统&#xff0c;如 MySQL、PostgreSQL、Oracle、SQL Server、SQLite、MongoDB 等。它提供了强大的 SQL 编辑和调试功能&#xff0c;使数据库开发和管理更加高…

DDR等长,到底长度差多少叫等长?

DDR4看这一篇就够了 - 知乎 (zhihu.com) 【全网首发】DDR4 PCB设计规范&设计要点PCB资源PCB联盟网 - Powered by Discuz! (pcbbar.com) 终于看到较为权威的DDR4等长要求了: !!!! 依据这个要求&#xff0c;H616项目的等长线不合格&#xff1a;

63、ELK安装和部署

一、ELK日志系统 1.1、ELK平台的定义 ELK平台是一套完整的日志集中处理解决方案&#xff0c;将ElasticSearch、Logstash和Kiabana 三个开源工具配合使用&#xff0c;完成更强大的用户对日志的查询、排序、统计需求 E:elasticsearch ES分布式索引型非关系数据库&#xff0c;存…

loguru日志模块:简化Python自动化测试的日志管理!

引言 日志是软件开发中的关键组成部分&#xff0c;为开发和测试人员提供了调试和监控应用程序的重要手段。loguru 是一个第三方的 Python 日志库&#xff0c;以其简洁的 API 和自动化的功能脱颖而出。本文将探讨为什么项目中需要日志&#xff0c;loguru 为何受到青睐&#xff…

书生大模型实战营-基础关-书生大模型全链路开源体系

书生大模型全链路开源体系 书生浦语大模型开源历程书生浦语2.0(InternLM2)体系模型到应用书生浦语开源开放体系 书生浦语大模型开源历程 书生浦语2.0(InternLM2)体系 书生浦语2.0体系&#xff0c;模型大小主要有2种规格&#xff1a; 7B&#xff0c;为轻量级的研究和应用提供了…

【面试】前端开发中的“八股文”:助力还是阻力?

引言 在程序员面试中&#xff0c;“八股文”已经成为一个不可或缺的环节。它通常指的是那些面试中频繁出现的、有固定答案的问题&#xff0c;涉及计算机科学的基础知识、编程语言的特性、以及一些常见的设计模式和算法。然而&#xff0c;围绕“八股文”的争议从未停歇。一方面…

系统设计中15 个最重要的权衡

系统设计的第一法则&#xff1a;一切都与权衡有关。 在设计系统时&#xff0c;我们需要决定要包含哪些功能以及要忽略哪些功能。每次我们做这个决定时&#xff0c;我们都在进行权衡。在本文中&#xff0c;我们将探讨系统设计中遇到的15个最常见的权衡问题&#xff0c;并使用实…

程序员转行大模型:从代码到无限可能

在技术日新月异的时代背景下&#xff0c;许多程序员开始思考自己的职业发展路径。面对着人工智能与机器学习领域的迅速崛起&#xff0c;越来越多的技术人员将目光投向了更为广阔的天地——转行成为大模型研究者或开发者。这一转变不仅要求个人技能的迭代升级&#xff0c;更是一…

数据库一张以时间排好序的表中,找出多次相邻的那些行,如何实现??

&#x1f3c6;本文收录于《CSDN问答解惑-专业版》专栏&#xff0c;主要记录项目实战过程中的Bug之前因后果及提供真实有效的解决方案&#xff0c;希望能够助你一臂之力&#xff0c;帮你早日登顶实现财富自由&#x1f680;&#xff1b;同时&#xff0c;欢迎大家关注&&收…

数据结构:线性表(上)

谈到线性的数据结构&#xff0c;那肯定离不开两个最基础的&#xff1a;数组和链表&#xff0c;当然有了数组和链表就会聊到栈和队列。 那么本篇我们就来介绍数组和链表 一、数组 数组&#xff08;Array&#xff09; 是一种很常见的数据结构。它由相同类型的元素&#xff08;…

分布式相关理论详解

目录 1.绪论 2.什么是分布式系统&#xff0c;和集群的区别 3.CAP理论 3.1 什么是CAP理论 3.2 一致性 3.2.1 计算机的一致性说明 1.事务中的一致性 2.并发场景下的一致性 3.分布式场景下的一致性 3.2.2 一致性分类 3.2.3 强一致性 1.线性一致性 a) 定义 a) Raft算法…

Java 并发编程:一文了解 Java 内存模型(处理器优化、指令重排序与内存屏障的深层解析)

大家好&#xff0c;我是栗筝i&#xff0c;这篇文章是我的 “栗筝i 的 Java 技术栈” 专栏的第 022 篇文章&#xff0c;在 “栗筝i 的 Java 技术栈” 这个专栏中我会持续为大家更新 Java 技术相关全套技术栈内容。专栏的主要目标是已经有一定 Java 开发经验&#xff0c;并希望进…

14.按钮和多选框

<p>爱好&#xff1a;<input type"checkbox" value"Riding" name"hobby">骑行<input type"checkbox" value"experiment" name"hobby">做实验<input type"checkbox" value"lea…