从C++编程入手设计模式——观察者模式

从C++编程入手设计模式——观察者模式

​ 观察者模式简直就是字如其名,观察观察,观察到了告诉别人。观察手的作用如此,观察者模式的工作机制也是如此。这个模式的核心思路是:一个对象的状态发生变化时,自动通知依赖它的其他对象,让它们自行更新

主要的组成部分

一是被观察者,也叫主题(Subject),它负责记录观察者并在自身状态发生变化时通知它们;换而言之,Subject作为信息的生成端,生成信息。

二是观察者(Observer),它主动注册到被观察者中,等待被通知,一旦收到消息就立即采取行动。这里生成端会使用远程的方式通知我们的被观察者投送信息。

​ 你可以把观察者模式想象成“订阅报纸”。报社就是主题,它维护一个订阅者名单(观察者列表)。每当有新一期的报纸(数据更新)出版时,它就向所有订阅者投递(通知)。每位订阅者收到报纸后根据自己的需要决定怎么处理,是直接阅读、剪报存档还是转发朋友。这种发布-订阅机制就是观察者模式的最基本形式。


在 C++ 中怎么写?

观察者模式的代码实现非常自然。我们会先定义一个抽象的观察者接口(通常叫 Observer 或者 Sender),让不同的具体观察者实现它;再定义一个主题类(通常叫 SubjectNotifier),它包含一个观察者列表,并负责注册、移除和通知观察者。每当主题内部数据变化(比如温度、湿度更新),它就会遍历观察者列表,挨个调用它们的“更新方法”。观察者模式的重点不在于通知的“数据内容”本身,而在于自动通知的机制


选择观察者模式?

在简单的程序中,我们可能会直接调用某些对象来执行逻辑。但如果程序复杂起来,比如某个数据更新后要引发十几种操作,硬编码这些调用就会让系统结构僵硬,维护困难。

观察者模式将“变化的主体”和“依赖的反应者”解耦。被观察者不用关心观察者是谁,它只管发出通知。观察者自己决定是否接收通知、如何响应。这让系统的扩展性和灵活性大大提高

比如天气系统中新加一个“语音播报设备”,只要实现观察者接口并注册进去,其他部分代码几乎无需改动。


观察者模式的几个关键词

  1. 低耦合:主题和观察者之间不互相依赖,只通过接口通信。
  2. 自动同步:当主题数据变化时,所有观察者自动得到通知。
  3. 一对多:一个主题可以对应多个观察者。
  4. 注册/注销机制:观察者可以自由加入或离开通知流程。

Example

​ 可以直接看一个代码框架来解决问题

#include <iostream>
#include <vector>
#include <memory>
#include <string>
#include <algorithm>// 观察者接口
class Observer {
public:virtual void update(const std::string& message) = 0;virtual ~Observer() = default;
};// 具体观察者
class ConcreteObserver : public Observer {
private:std::string name;
public:explicit ConcreteObserver(std::string n) : name(std::move(n)) {}void update(const std::string& message) override {std::cout << "Observer [" << name << "] received: " << message << '\n';}
};// 主题接口
class Subject {
public:virtual void attach(std::shared_ptr<Observer> observer) = 0;virtual void detach(std::shared_ptr<Observer> observer) = 0;virtual void notify(const std::string& message) = 0;virtual ~Subject() = default;
};// 具体主题
class ConcreteSubject : public Subject {
private:std::vector<std::shared_ptr<Observer>> observers;
public:void attach(std::shared_ptr<Observer> observer) override {observers.push_back(std::move(observer));}void detach(std::shared_ptr<Observer> observer) override {observers.erase(std::remove(observers.begin(), observers.end(), observer), observers.end());}void notify(const std::string& message) override {for (const auto& observer : observers) {observer->update(message);}}
};int main() {auto subject = std::make_shared<ConcreteSubject>();auto obs1 = std::make_shared<ConcreteObserver>("Observer1");auto obs2 = std::make_shared<ConcreteObserver>("Observer2");subject->attach(obs1);subject->attach(obs2);subject->notify("Event 1 occurred");subject->detach(obs1);subject->notify("Event 2 occurred");return 0;
}

​ 你可以使用这样的方式来作为事件驱动框架的一个根基。

练习题

天气站系统

要求:设计一个天气数据发布系统,天气站(WeatherStation)是“主题”,可通知多个“观察者”显示设备(如手机、网页等)。

功能要求

  1. 观察者可以注册/注销。
  2. 当温度或湿度变化时,所有注册观察者会收到更新并打印数据。(注:这个部分笔者的参考代码没有实现,您自己酌情思考)

提示

  • 使用接口 IObserver 表示观察者,定义如 update(float temperature, float humidity)
  • 主题接口 ISubject 需要支持 registerObserver()removeObserver()notifyObservers()
  • 可以实现两个观察者类 PhoneDisplayWebDisplay

modern-cpp-patterns-playground/Observer/WeatherForecast at main · Charliechen114514/modern-cpp-patterns-playground

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

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

相关文章

MITM 中间人攻击

​据Akamai 2023网络安全报告显示&#xff0c;MITM攻击在数据泄露事件中占比达32.7%&#xff0c;平均每次事件造成企业损失$380,000​ ​NIST研究指出&#xff1a;2022-2023年高级MITM攻击增长41%&#xff0c;近70%针对金融和医疗行业​ 一、MITM攻击核心原理与技术演进 1. 中…

llama_index chromadb实现RAG的简单应用

此demo是自己提的一个需求&#xff1a;用modelscope下载的本地大模型实现RAG应用。毕竟大模型本地化有利于微调&#xff0c;RAG使内容更有依据。 为什么要用RAG&#xff1f; 由于大模型存在一定的局限性&#xff1a;知识时效性不足、专业领域覆盖有限以及生成结果易出现“幻觉…

TDMQ CKafka 版事务:分布式环境下的消息一致性保障

解锁 CKafka 事务能力的神秘面纱 在当今数字化浪潮下&#xff0c;分布式系统已成为支撑海量数据处理和高并发业务的中流砥柱。但在这看似坚不可摧的架构背后&#xff0c;数据一致性问题却如影随形&#xff0c;时刻考验着系统的稳定性与可靠性。 CKafka 作为分布式流处理平台的…

常见的负载均衡算法

常见的负载均衡算法 在实现水平扩展过程中&#xff0c;负载均衡算法是决定请求如何在多个服务实例间分配的核心逻辑。一个合理的负载均衡策略能够有效分散系统压力&#xff0c;提升系统吞吐能力与稳定性。 负载均衡算法可部署在多种层级中&#xff0c;如七层HTTP反向代理&…

数据结构转换与离散点生成

在 C 开发中&#xff0c;我们常常需要在不同的数据结构之间进行转换&#xff0c;以满足特定库或框架的要求。本文将探讨如何将 std::vector<gp_Pnt> 转换为 QVector<QPointF>&#xff0c;并生成特定范围内的二维离散点。 生成二维离散点 我们首先需要生成一系列…

零基础学习Redis(12) -- Java连接redis服务器

在我们之前的内容中&#xff0c;我们会发现通过命令行操作redis是十分不科学的&#xff0c;所以redis官方提供了redis的应用层协议RESP&#xff0c;更具这个协议可以实现一个和redis服务器通信的客户端程序&#xff0c;来简化和完善redis的使用。现阶段有很多封装了RESP协议的库…

clangd LSP 不能找到项目中的文件

clangd LSP 不能找到项目中的文件 clangd LSP 不能找到项目中的文件 clangd LSP 不能找到项目中的文件 Normally you need to create compile_commands.json。 如果你使用 cmake 作为构建工具&#xff0c;请执行下面的命令&#xff1a; cmake -DCMAKE_EXPORT_COMPILE_COMMAN…

【内存】Linux 内核优化实战 - vm.overcommit_memory

目录 vm.overcommit_memory 解释一、概念与作用二、参数取值与含义三、相关参数与配置方式四、实际应用场景建议五、注意事项 vm.overcommit_memory 解释 一、概念与作用 vm.overcommit_memory 是 Linux 内核中的一个参数&#xff0c;用于控制内存分配的“过度承诺”&#xf…

Python:.py文件转换为双击可执行的Windows程序(版本2)

流程步骤&#xff1a; 这个流程图展示了将 Python .py 文件转换为 Windows 可执行程序的完整过程&#xff0c;主要包括以下步骤&#xff1a; 1、准备 Python文件&#xff0c;确保代码可独立运行 2、安装打包工具&#xff08;如 PyInstaller&#xff09; 3、打开命令提示符并定位…

【请关注】mysql一些经常用到的高级SQL

经常去重复数据&#xff0c;数据需要转等操作&#xff0c;汇总高级SQL MySQL操作 一、数据去重&#xff08;Data Deduplication&#xff09; 去重常用于清除重复记录&#xff0c;保留唯一数据。 1. 使用DISTINCT关键字去重单列 -- 从用户表中获取唯一的邮箱地址 SELECT DISTIN…

RA4M2开发涂鸦模块CBU(2)----配置按键开启LED

RA4M2开发涂鸦模块CBU.2--配置按键开启LED 概述视频教学样品申请硬件准备参考程序按键口配置中断回调函数主程序 概述 本实验演示如何在 Renesas RA4M2 单片机上使用 GPIO 输入&#xff08;按键&#xff09; 触发 GPIO 输出&#xff08;LED&#xff09;&#xff0c;并使用e2st…

Linux——Json

一 概念 json是一种轻量级&#xff0c;基于文本的&#xff0c;可读的数据交换格式&#xff0c;能够让数据在不同系统&#xff08;比如前端—后端&#xff0c;服务器—客户端&#xff09;间方便传递/存储。在编程语言中都内置了处理json数据的方法 二 语法规则 1. 数据格式&a…

大模型之微调篇——指令微调数据集准备

写在前面 高质量数据的准备是微调大模型的重中之重&#xff0c;一些高质量的数据集可能远比模型性能更佳重要。 我是根据自己的数据照着B站up code花园LLaMA Factory 微调教程&#xff1a;如何构建高质量数据集&#xff1f;_哔哩哔哩_bilibili做的。 数据集格式 在LLaMA Fa…

LVS—DR模式

LVS—DR模式 LVS DR 模式详细简介 一、模式定义与核心原理 LVS DR&#xff08;Direct Routing&#xff09;模式&#xff0c;即直接路由模式&#xff0c;是 Linux Virtual Server&#xff08;LVS&#xff09;实现负载均衡的经典模式之一&#xff0c;工作于网络四层&#xff0…

宝玉分享VibeCoding构建Agent

借助 Claude Code 完成的一个翻译智能体 (Translator Agent)。你只需输入一段文字、一个网址或一个本地文件路径&#xff0c;它就能自动提取内容并完成翻译。更酷的是&#xff0c;它还能修正原文中的拼写错误&#xff0c;确保译文的准确流畅。 到底什么是“真正的”AI Agent&a…

在spring boot中使用Logback

在 Spring Boot 中使用 Logback 作为日志框架是开发中的常见需求&#xff0c;因其高性能和灵活配置而广受青睐。以下是详细实践指南&#xff0c;结合了配置方法、代码示例及最佳实践&#xff1a; &#x1f527; 一、依赖配置 Spring Boot 默认集成了 Logback&#xff0c;无需手…

腾讯云 Lighthouse 轻量应用服务器:数据驱动的架构选型指南

摘要&#xff1a;腾讯云 Lighthouse 作为面向轻量级应用场景的优化解决方案&#xff0c;通过高性价比套餐式售卖、开箱即用应用模板及流量包计费模式&#xff0c;显著降低中小企业与开发者的上云门槛。本文基于性能测试与横向对比&#xff0c;量化分析其核心优势与适用边界。 …

Linux TCP/IP协议栈中的TCP输入处理:net/ipv4/tcp_input.c解析

在网络通信领域,TCP(传输控制协议)因其可靠的面向连接特性而被广泛应用。Linux内核的TCP/IP协议栈实现了对TCP协议的高效处理,其中net/ipv4/tcp_input.c文件扮演着关键角色,负责处理TCP数据包的输入逻辑。下面是对该文件核心功能的深入剖析。 一、TCP数据包接收与处理 (…

物联网传输网关、RTU、DTU及SCADA系统技术解析

目录 摘要 一、引言 二、物联网传输网关 1. 定义 2. 类型 3. 分类 4. 工作原理 5. 差异分析 总结&#xff1a; 三、RTU&#xff08;远程终端单元&#xff09; 1. 定义 2. 工作原理 3. 特点 4. 应用场景 四、DTU&#xff08;数据传输单元&#xff09; 1. 定义 …

【unity游戏开发——热更新】YooAsset简化资源加载、打包、更新等流程

注意&#xff1a;考虑到热更新的内容比较多&#xff0c;我将热更新的内容分开&#xff0c;并全部整合放在【unity游戏开发——热更新】专栏里&#xff0c;感兴趣的小伙伴可以前往逐一查看学习。 文章目录 前言1、什么是YooAsset&#xff1f;2、系统需求3、系统特点 一、下载安装…