突破编程_C++_设计模式(命令模式)
1 命令模式的基本概念
C++ 命令模式是一种设计模式,它允许将请求封装为一个对象,从而可以用不同的请求对客户进行参数化;对请求排队或记录请求日志,以及支持可撤销的操作。命令模式的主要目的是将请求封装为对象,从而可以使用不同的请求把客户端与接收者解耦。
在命令模式中,通常包含以下几个关键角色:
(1)命令(Command): 这是一个抽象类,它声明了执行操作的接口。具体的命令类会实现这个接口,并绑定到一个接收者对象上。当命令对象被调用时,它会调用接收者的相应操作。
(2)具体命令(ConcreteCommand): 这是命令接口的具体实现类,它持有一个接收者对象,并实现了命令接口中的执行方法。当执行方法被调用时,它会调用接收者的相应方法来完成请求的操作。
(3)请求者(Invoker): 请求者负责调用命令对象的执行方法。它不需要知道具体的命令对象类型,只需要知道它是一个命令对象即可。
(4)接收者(Receiver): 接收者对象知道如何执行请求的操作。它具体实现了请求的功能。
2 命令模式的实现步骤
命令模式的实现步骤如下:
(1)定义命令接口:
- 创建一个抽象类(或接口),声明一个执行操作的方法。这个方法将是所有具体命令类共有的接口。
(2)实现具体命令类:
- 创建具体命令类,继承自命令接口。
- 在具体命令类中,通常包含一个指向接收者对象的指针(或引用)。
- 实现执行操作的方法,该方法内部调用接收者对象的相应方法来完成请求的操作。
(3)创建接收者类:
- 定义一个接收者类,它包含实际执行操作的逻辑。
- 接收者类的方法对应于命令类需要执行的操作。
(4)创建请求者类:
- 请求者类负责调用命令对象的执行方法。
- 请求者不需要知道具体的命令对象类型,它只需要持有一个命令对象的指针(或引用)。
- 请求者类可以存储多个命令对象,并按照需要调用它们的执行方法。
(5)将命令对象与接收者对象关联:
- 在创建具体命令对象时,将接收者对象传递给命令对象,以便命令对象在执行时可以调用接收者的方法。
(6)使用命令模式:
- 在客户端代码中,创建接收者对象。
- 创建与接收者对象关联的具体命令对象。
- 将命令对象传递给请求者对象。
- 请求者对象调用命令对象的执行方法,从而间接调用接收者的方法,完成请求的操作。
如下为样例代码:
#include <iostream>
#include <memory>
#include <string>
#include <vector> // 创建接收者类
class Receiver {
public:void action() {std::cout << "Receiver::action() called" << std::endl;}
};// 定义命令接口
class Command {
public:virtual ~Command() = default;virtual void execute() = 0;
};// 实现具体命令类
class ConcreteCommand : public Command {
public:ConcreteCommand(std::shared_ptr<Receiver> receiver) : m_receiver(receiver) {}void execute() override {m_receiver->action();}
private:std::shared_ptr<Receiver> m_receiver;
};// 创建请求者类
class Invoker {
public:void setCommand(std::unique_ptr<Command> command) {commands.push_back(std::move(command));}void executeCommands() {for (auto& cmd : commands) {cmd->execute();}}private:std::vector<std::unique_ptr<Command>> commands;
};// 客户端代码
int main()
{// 创建接收者对象 std::shared_ptr<Receiver> receiver = std::make_shared<Receiver>();// 创建具体命令对象,并与接收者对象关联 std::unique_ptr<Command> command = std::make_unique<ConcreteCommand>(receiver);// 创建请求者对象,并将命令对象传递给请求者 Invoker invoker;invoker.setCommand(std::move(command));// 请求者调用命令对象的执行方法 invoker.executeCommands(); // 输出: Receiver::action() called return 0;
}
上面代码的输出为:
Receiver::action() called
在这个示例中:
- Command 是一个抽象类,定义了命令的接口。
- ConcreteCommand 是 Command 的具体实现,它持有一个指向 Receiver 的 shared_ptr,并在 execute 方法中调用 Receiver 的 action 方法。
- Receiver 是接收者类,包含了实际要执行的操作。
- Invoker 是请求者类,它持有一个 Command 对象的列表,并提供了一个方法来执行这些命令。
在 main 函数中,创建了 Receiver 对象和 ConcreteCommand 对象,并将 Receiver 对象传递给 ConcreteCommand 对象。然后将 ConcreteCommand 对象传递给 Invoker 对象,并调用 Invoker 的 executeCommands 方法来执行命令。最终,Receiver 的 action 方法被调用,输出了相应的信息。
3 命令模式的应用场景
C++命令模式的应用场景广泛,主要适用于以下情况:
(1)当功能需要支持撤销和恢复撤销操作时: 命令模式通过封装命令对象,可以方便地记录和执行命令历史,从而支持撤销和恢复撤销操作。这在图形编辑、文本编辑等应用中非常有用,例如,在绘制图形时,用户可以撤销最近的操作以返回到之前的状态。
(2)当需要设计一组命令,并且命令之间可以相互组合时: 命令模式支持将多个命令组合成一个复合命令,从而可以一次性执行多个操作。这在需要执行一系列相关操作时非常有用,例如,在编辑器中,用户可能希望将多个编辑操作(如剪切、复制、粘贴)组合成一个宏命令,以便一次性执行。
(3)当系统需要将命令发起者和命令执行者解耦时: 命令模式允许将请求封装为命令对象,从而使发送请求的对象和执行请求的对象解耦。这使得发送请求的对象不需要知道请求如何被完成,增加了系统的灵活性和可维护性。例如,在用户界面和后台逻辑之间,命令模式可以确保它们之间的松耦合,使得修改其中一方不会影响到另一方。
(4)当系统需要在不同时间指定、排列和执行请求时: 命令模式允许将命令对象存储在队列或列表中,然后根据需要按顺序执行它们。这对于实现批量操作、延迟执行或任务调度等功能非常有用。
总的来说,C++命令模式适用于需要解耦请求发送者和请求执行者、支持撤销操作、需要组合多个命令或在不同时间执行请求的场景。它能够提高系统的灵活性、可维护性和可扩展性。
3.1 命令模式应用于需要支持撤销和恢复撤销操作的场景
在 C++ 中,命令模式特别适用于需要支持撤销和恢复撤销操作的场景,如下为样例代码:
#include <iostream>
#include <memory>
#include <vector>
#include <stack>
#include <string> // 接收者类
class Receiver {
public:std::string action() {std::string currentState = "New state"; // 假设这是某种状态 // 执行一些操作... return currentState;}void restore(const std::string& state) {// 恢复到之前的状态... std::cout << "Restoring to state: " << state << std::endl;}
};// 命令接口
class Command {
public:virtual ~Command() = default;virtual void execute() = 0;virtual void undo() = 0;
};// 具体命令类
class ConcreteCommand : public Command {
public:ConcreteCommand(std::shared_ptr<Receiver> receiver) : receiver(receiver) {}void execute() override {previousState = receiver->action();std::cout << "Command executed: " << previousState << std::endl;}void undo() override {receiver->restore(previousState);std::cout << "Command undone: " << previousState << std::endl;}private:std::shared_ptr<Receiver> receiver;std::string previousState;
};// 撤销管理器
class UndoManager {
public:void executeCommand(std::shared_ptr<Command> cmd) {cmd->execute();commands.push(cmd);}void undoLastCommand() {if (!commands.empty()) {auto cmd = commands.top();commands.pop();cmd->undo();}else {std::cout << "No commands to undo." << std::endl;}}private:std::stack<std::shared_ptr<Command>> commands;
};int main()
{// 创建接收者对象 auto receiver = std::make_shared<Receiver>();// 创建命令对象 auto command = std::make_shared<ConcreteCommand>(receiver);// 创建撤销管理器 UndoManager undoManager;// 执行命令 undoManager.executeCommand(command);// 撤销命令 undoManager.undoLastCommand();return 0;
}
上面代码的输出为:
Command executed: New state
Restoring to state: New state
Command undone: New state
在这个示例中:
- Command 是一个抽象类,定义了命令的接口,包括 execute 和 undo 方法。
- ConcreteCommand 是 Command 的具体实现,它持有一个指向 Receiver 的 shared_ptr,并在 execute 方法中执行操作,同时保存当前状态以便在 undo 方法中恢复。
- Receiver 类包含实际执行操作的逻辑,以及一个 restore 方法用于恢复之前的状态。
- UndoManager 类负责管理命令的撤销操作,它使用一个 stack 来存储执行过的命令对象。当调用 executeCommand 方法时,它会执行命令并将其推入栈中;当调用 undoLastCommand 方法时,它会从栈顶弹出命令并执行其 undo 方法。
在 main 函数中,创建了一个 Receiver 对象和一个 ConcreteCommand 对象,并通过 UndoManager 来执行和撤销命令。通过使用 shared_ptr,可以确保 Receiver 和 ConcreteCommand 对象的生命周期得到妥善管理,即使在命令被撤销后也不会导致内存泄漏。
3.2 命令模式应用需要设计一组命令,并且命令之间可以相互组合的场景
在C++中,命令模式可以用于设计一组命令,并允许这些命令之间可以相互组合,形成复合命令。当需要执行一系列操作作为一个单一命令时,这种模式特别有用。如下为样例代码:
#include <iostream>
#include <memory>
#include <vector> // 命令接口
class Command {
public:virtual ~Command() = default;virtual void execute() = 0;
};// 具体命令类A
class CommandA : public Command {
public:void execute() override {std::cout << "Executing CommandA" << std::endl;}
};// 具体命令类B
class CommandB : public Command {
public:void execute() override {std::cout << "Executing CommandB" << std::endl;}
};// 复合命令类
class CompositeCommand : public Command {
public:void addCommand(std::shared_ptr<Command> cmd) {commands.push_back(cmd);}void execute() override {for (const auto& cmd : commands) {cmd->execute();}}private:std::vector<std::shared_ptr<Command>> commands;
};int main()
{// 创建具体命令对象 auto cmdA = std::make_shared<CommandA>();auto cmdB = std::make_shared<CommandB>();// 创建复合命令对象 auto compositeCmd = std::make_shared<CompositeCommand>();// 将具体命令添加到复合命令中 compositeCmd->addCommand(cmdA);compositeCmd->addCommand(cmdB);// 执行复合命令 compositeCmd->execute();return 0;
}
上面代码的输出为:
Executing CommandA
Executing CommandB
在这个示例中:
- Command 是一个抽象基类,定义了命令的接口,即 execute 方法。
- CommandA 和 CommandB 是具体命令类,分别实现了 execute 方法以执行特定的操作。
- CompositeCommand 是一个复合命令类,它包含一个命令对象的 vector 容器。通过 addCommand 方法,可以将多个具体命令或复合命令添加到复合命令中。当调用 execute 方法时,它会依次执行容器中的每个命令。
在 main 函数中,创建了两个具体命令对象 cmdA 和 cmdB,然后创建了一个复合命令对象 compositeCmd。接着,将 cmdA 和 cmdB 添加到 compositeCmd 中。最后,调用 compositeCmd 的 execute 方法来执行复合命令,这将依次执行 cmdA 和 cmdB。
4 命令模式的优点与缺点
C++ 命令模式的优点主要包括:
(1)封装性: 每个命令都被封装起来,客户端无需知道命令具体是怎么执行的,只需调用相应的命令。这使得命令模式能够很好地隐藏实现细节,使系统更加模块化和可维护。
(2)扩展性: 命令模式符合开闭原则,即对扩展开放,对修改关闭。这意味着可以轻松地添加新命令而无需修改现有代码。
(3)支持撤销和重做: 命令模式可以方便地实现对请求的撤销和重做,这对于需要支持撤销操作的系统来说非常有用。
(4)请求排队和记录: 命令模式可以较容易地设计一个命令队列,以及在需要的情况下将命令记入日志。
(5)解耦: 命令模式将请求一个操作的对象与知道如何执行一个操作的对象分隔开,降低了系统的耦合度。
然而,C++ 命令模式也存在一些缺点:
(1)可能产生过多的具体命令类: 对于每一个命令,都需要设计一个具体命令类,这可能导致系统中存在大量的具体命令类,增加了系统的复杂性。
(2)可能导致过度设计: 在一些简单的场景下,使用命令模式可能过于复杂,导致过度设计,增加不必要的开销。
相关文章:
突破编程_C++_设计模式(命令模式)
1 命令模式的基本概念 C 命令模式是一种设计模式,它允许将请求封装为一个对象,从而可以用不同的请求对客户进行参数化;对请求排队或记录请求日志,以及支持可撤销的操作。命令模式的主要目的是将请求封装为对象,从而可…...
LeetCode102题:二叉树的层序遍历(python3)
代码思路:使用队列先进先出的特性,queue[]不为空进入for循环,tmp存储每层的节点,将结果添加至res[]中。 python中使用collections中的双端队列deque(),其popleft()方法可达到O(1)时间复杂度。 class Solution:def lev…...
linux服务器保存git账号密码命令
1.保存git账号密码 git config --global credential.helper store 2.查看git账号密码 cd回车 ls -a cat .git-credentials 步骤: 先输入 git config --global credential.helper store 然后进入代码目录,git pull 会提示输入git账号、密码,因为我们提前输…...
基于YOLOv8/YOLOv7/YOLOv6/YOLOv5的田间杂草检测系统(深度学习模型+UI界面+Python代码+训练数据集)
摘要:开发用于田间杂草识别的系统对提高农业运营效率和提升作物产出至关重要。本篇文章详尽阐述了如何应用深度学习技术开发一个用于田间杂草识别的系统,并附上了完备的代码实现。该系统基于先进的YOLOv8算法,并对比了YOLOv7、YOLOv6、YOLOv5…...
java Lambda表达式如何支持静态方法引用
java Lambda表达式如何支持静态方法引用 在Java中,Lambda表达式支持静态方法引用,允许你直接使用静态方法作为Lambda表达式的实现。静态方法引用使用类名和方法名来引用静态方法。 下面是一个简单的示例,展示了如何在Lambda表达式中使用静态…...
SpringMVC04、Controller 及 RestFul
4、Controller 及 RestFul 4.1、控制器Controller 控制器复杂提供访问应用程序的行为,通常通过接口定义或注解定义两种方法实现。控制器负责解析用户的请求并将其转换为一个模型。在Spring MVC中一个控制器类可以包含多个方法在Spring MVC中,对于Contr…...
【机器学习300问】33、决策树是如何进行特征选择的?
还记得我在【机器学习300问】的第28问里谈到的,看决策树的定义不就是if-else语句吗怎么被称为机器学习模型?其中最重要的两点就是决策树算法要能够自己回答下面两问题: 该选哪些特征 特征选择该选哪个阈值 阈值确定 今天这篇文章承接上文&…...
剑指offer C ++双栈实现队列
1. 基础 队列:先进先出,即插入数据在队尾进行,删除数据在队头进行; 栈:后进先出,即插入与删除数据均在栈顶进行。 2. 思路 两个栈实现一个队列的思想:用pushStack栈作为push数据的栈ÿ…...
【YOLOv9】训练模型权重 YOLOv9.pt 重新参数化轻量转为 YOLOv9-converted.pt
【YOLOv9】训练模型权重 YOLOv9.pt 重新参数化轻量转为 YOLOv9-converted.pt 1. 模型权重准备2. 模型重新参数化2.1 文件准备2.2 参数修改2.3 重新参数化过程 3. 重新参数化后模型推理3.1 推理超参数配置3.2 模型推理及对比 4. onnx 模型导出(补充内容)4…...
Zookeeper搭建
目录 前言 初了解Zookeeper 搭建 准备 配置Zookeeper 前言 今天来介绍Zookeeper的搭建,其实Zookeeper的搭建很简单,但是为什么还要单独整一节呢,这就不得不先了解Zookeeper有什么功能了!而且现在很火的框架也离不开Zookeepe…...
2.Datax数据同步之Windows下,mysql和sqlserver之间的自定义sql文数据同步
目录 前言步骤操作大纲步骤明细mysql 至 sqlServersqlServer 至 mysql执行同步语句中报 前言 上一篇文章实现了不同的mysql数据库之间的数据同步,在此基础上本篇将实现mysql和sqlserver之间的自定义sql文数据同步 准备工作: JDK(1.8以上,推…...
commonjs和esmodule
commonjs的模块导出和引用写法: lib.js 导出一个模块 let a 1 let b 2 function aPlus1() {return a } module.exports {a,b,aPlus1 } index.js引用一个模块 const {a,b,aPlus1} require(./lib.js) console.log(hh:,a) esmodule的模块导出和引用方法&#x…...
Android的编译系统
安卓的编译真的太多吐槽的地方了,有必须到croot下编译的,有随便改个.c就要七八分钟编译的。我有时候真的不知道这么多开发人员是怎么挺过来的。 今晚简单看看这个编译系统soong吧。 算了,下面这个写的很好了,我先看看吧。。。 …...
Midjourney指控Stability AI夜袭数据,网络风波一触即发
每周跟踪AI热点新闻动向和震撼发展 想要探索生成式人工智能的前沿进展吗?订阅我们的简报,深入解析最新的技术突破、实际应用案例和未来的趋势。与全球数同行一同,从行业内部的深度分析和实用指南中受益。不要错过这个机会,成为AI领…...
JVM知识整体学习
前言:本篇没有任何建设性的想法,只是我很早之前在学JVM时记录的笔记,只是想从个人网站迁移过来。文章其实就是对《深入理解JVM虚拟机》的提炼,纯基础知识,网上一搜一大堆。 一、知识点脑图 本文只谈论HotSpots虚拟机。…...
蓝桥杯--日期统计
目录 一、题目 二、解决代码 三、代码分析 四、另一种思路 五、关于set文章推荐 一、题目 二、解决代码 #include <bits/stdc.h> using namespace std; int main() {int arr[100] { 5,6,8,6,9,1,6,1,2,4,9,1,9,8,2,3,6,4,7,7,5,9,5,0,3,8,7,5,8,1,5,8,6,1,8,3,0,…...
[leetcode~dfs]1261. 在受污染的二叉树中查找元素
给出一个满足下述规则的二叉树: root.val 0 如果 treeNode.val x 且 treeNode.left ! null,那么 treeNode.left.val 2 * x 1 如果 treeNode.val x 且 treeNode.right ! null,那么 treeNode.right.val 2 * x 2 现在这个二叉树受到「污…...
PyQt5使用
安装Pyqt5信号与槽使用可视化界面编辑UI (Pyside2)ui生成之后的使用(两种方法)1 ui转化为py文件 进行import2 动态调用UI文件 安装Pyqt5 pip install pyqt5-tools这时候我们使用纯代码实现一个简单的界面 from PyQt5.QtWidgets import QApplication, QMainWindow, QPushButto…...
利用GPT开发应用005:Codex、Turbo、ChatGPT、GPT-4
文章目录 一、GPT-3 Codex二、GPT-3.5 Turbo二、ChatGPT三、GPT-4 一、GPT-3 Codex 2022年3月,OpenAI 发布了 GPT-3 Codex 的新版本。 这个新模型具有编辑和插入文本的能力。它们是通过截至 2021 年 6 月的数据进行训练的,并被描述为比之前版本更强大。到…...
制造行业大数据应用:四大领域驱动产业升级与智慧发展
一、大数据应用:制造行业的智慧引擎 随着大数据技术的不断突破与普及,制造行业正迎来一场前所未有的变革。大数据应用,如同智慧引擎一般,为制造行业注入了新的活力,推动了产业升级与创新发展。 二、大数据应用在制造行…...
【Oracle APEX开发小技巧12】
有如下需求: 有一个问题反馈页面,要实现在apex页面展示能直观看到反馈时间超过7天未处理的数据,方便管理员及时处理反馈。 我的方法:直接将逻辑写在SQL中,这样可以直接在页面展示 完整代码: SELECTSF.FE…...
Redis相关知识总结(缓存雪崩,缓存穿透,缓存击穿,Redis实现分布式锁,如何保持数据库和缓存一致)
文章目录 1.什么是Redis?2.为什么要使用redis作为mysql的缓存?3.什么是缓存雪崩、缓存穿透、缓存击穿?3.1缓存雪崩3.1.1 大量缓存同时过期3.1.2 Redis宕机 3.2 缓存击穿3.3 缓存穿透3.4 总结 4. 数据库和缓存如何保持一致性5. Redis实现分布式…...
遍历 Map 类型集合的方法汇总
1 方法一 先用方法 keySet() 获取集合中的所有键。再通过 gey(key) 方法用对应键获取值 import java.util.HashMap; import java.util.Set;public class Test {public static void main(String[] args) {HashMap hashMap new HashMap();hashMap.put("语文",99);has…...
3.3.1_1 检错编码(奇偶校验码)
从这节课开始,我们会探讨数据链路层的差错控制功能,差错控制功能的主要目标是要发现并且解决一个帧内部的位错误,我们需要使用特殊的编码技术去发现帧内部的位错误,当我们发现位错误之后,通常来说有两种解决方案。第一…...
基于uniapp+WebSocket实现聊天对话、消息监听、消息推送、聊天室等功能,多端兼容
基于 UniApp + WebSocket实现多端兼容的实时通讯系统,涵盖WebSocket连接建立、消息收发机制、多端兼容性配置、消息实时监听等功能,适配微信小程序、H5、Android、iOS等终端 目录 技术选型分析WebSocket协议优势UniApp跨平台特性WebSocket 基础实现连接管理消息收发连接…...
CMake基础:构建流程详解
目录 1.CMake构建过程的基本流程 2.CMake构建的具体步骤 2.1.创建构建目录 2.2.使用 CMake 生成构建文件 2.3.编译和构建 2.4.清理构建文件 2.5.重新配置和构建 3.跨平台构建示例 4.工具链与交叉编译 5.CMake构建后的项目结构解析 5.1.CMake构建后的目录结构 5.2.构…...
2021-03-15 iview一些问题
1.iview 在使用tree组件时,发现没有set类的方法,只有get,那么要改变tree值,只能遍历treeData,递归修改treeData的checked,发现无法更改,原因在于check模式下,子元素的勾选状态跟父节…...
Neo4j 集群管理:原理、技术与最佳实践深度解析
Neo4j 的集群技术是其企业级高可用性、可扩展性和容错能力的核心。通过深入分析官方文档,本文将系统阐述其集群管理的核心原理、关键技术、实用技巧和行业最佳实践。 Neo4j 的 Causal Clustering 架构提供了一个强大而灵活的基石,用于构建高可用、可扩展且一致的图数据库服务…...
全面解析各类VPN技术:GRE、IPsec、L2TP、SSL与MPLS VPN对比
目录 引言 VPN技术概述 GRE VPN 3.1 GRE封装结构 3.2 GRE的应用场景 GRE over IPsec 4.1 GRE over IPsec封装结构 4.2 为什么使用GRE over IPsec? IPsec VPN 5.1 IPsec传输模式(Transport Mode) 5.2 IPsec隧道模式(Tunne…...
使用 SymPy 进行向量和矩阵的高级操作
在科学计算和工程领域,向量和矩阵操作是解决问题的核心技能之一。Python 的 SymPy 库提供了强大的符号计算功能,能够高效地处理向量和矩阵的各种操作。本文将深入探讨如何使用 SymPy 进行向量和矩阵的创建、合并以及维度拓展等操作,并通过具体…...
