当前位置: 首页 > news >正文

《设计模式》命令模式

《设计模式》命令模式

命令模式(Command Pattern)是一种行为型设计模式,它将请求和处理分开,使得请求发送者和接收者解耦,从而降低系统的耦合度。在命令模式中,请求被封装为一个独立的对象,并且将其参数化,以便在不同的请求中传递不同的参数。命令模式中的对象包括:

  1. 请求者(Invoker):负责向命令对象发起请求,并将命令对象设置为接收者的命令。
  2. 命令(Command):封装了一次请求,包括请求的接收者和对接收者的操作。命令对象可以执行操作、撤销操作和恢复操作。
  3. 接收者(Receiver):真正执行命令操作的对象。接收者知道如何实现命令所要求的操作。
  4. 客户端(Client):创建请求者、命令和接收者对象,并将它们组装起来。

命令模式的基本思想是将请求封装为一个对象,从而使请求的发送者和接收者之间解耦。通过将请求封装成命令对象,可以将请求的调用者与实现者分离开来,从而实现请求者和接收者之间的解耦。同时,由于命令对象可以被序列化和存储,因此可以支持撤销、重做、事务等功能。

在命令模式中,可以通过多种方式来实现命令的撤销和重做。例如,可以使用栈来保存每个命令的执行历史记录,当需要撤销命令时,从栈中弹出最近的一个命令并执行撤销操作;当需要重做命令时,从栈中弹出最近的一个撤销命令并执行恢复操作。另外,还可以使用备忘录模式来实现命令的撤销和重做。

命令模式的优点包括:

  1. 降低系统的耦合度。将命令的请求者和执行者分离,使得它们不需要了解彼此的实现细节,从而降低了系统的耦合度,提高了系统的可维护性和可扩展性。
  2. 易于扩展新的命令。由于命令模式将每个命令封装为一个独立的对象,因此很容易添加新的命令,只需要增加新的命令类即可。
  3. 支持撤销和重做操作。由于每个命令对象都保存了执行历史记录,因此可以通过撤销和重做操作来实现对命令的撤销和恢复。

命令模式适用于以下场景:

  1. 当需要将命令的请求者和执行者解耦时,可以使用命令模式。
  2. 当需要支持撤销和重做操作时,可以使用命令模式。
  3. 当需要支持宏命令(即由多个命令组成的一个命令)时,可以使用命令模式。
  4. 当需要支持命令的排队、记录日志、事务等操作时,可以使用命令模式。例如,在数据库系统中,可以使用命令模式来实现事务操作。

总之,命令模式可以有效地解耦命令的请求者和执行者,提高系统的可维护性和可扩展性。

#include <iostream>
#include <string>
#include <vector>// 接收者类,执行具体的操作
class Receiver {
public:void action1() {std::cout << "接收者执行操作1" << std::endl;}void action2() {std::cout << "接收者执行操作2" << std::endl;}
};// 命令接口
class Command {
public:virtual ~Command() {}virtual void execute() = 0;
};// 具体命令类,实现命令接口
class ConcreteCommand1 : public Command {
public:ConcreteCommand1(Receiver* receiver) : m_receiver(receiver) {}virtual ~ConcreteCommand1() {}virtual void execute() {m_receiver->action1();}private:Receiver* m_receiver;
};class ConcreteCommand2 : public Command {
public:ConcreteCommand2(Receiver* receiver) : m_receiver(receiver) {}virtual ~ConcreteCommand2() {}virtual void execute() {m_receiver->action2();}private:Receiver* m_receiver;
};// 宏命令类,由多个命令组成
class MacroCommand : public Command {
public:MacroCommand() {}virtual ~MacroCommand() {}void addCommand(Command* cmd) {m_cmds.push_back(cmd);}virtual void execute() {for (auto cmd : m_cmds) {cmd->execute();}}private:std::vector<Command*> m_cmds;
};// 命令发起者
class Invoker {
public:Invoker() {}~Invoker() {}void setCommand(Command* cmd) {m_cmd = cmd;}void executeCommand() {m_cmd->execute();}private:Command* m_cmd;
};// 客户端代码
int main() {Receiver* receiver = new Receiver();Command* cmd1 = new ConcreteCommand1(receiver);Command* cmd2 = new ConcreteCommand2(receiver);MacroCommand* macroCmd = new MacroCommand();macroCmd->addCommand(cmd1);macroCmd->addCommand(cmd2);Invoker invoker;invoker.setCommand(macroCmd);invoker.executeCommand();delete cmd1;delete cmd2;delete macroCmd;delete receiver;return 0;
}

在上面的示例中,我们定义了一个接收者类 Receiver,它负责执行具体的操作。然后,我们定义了两个具体命令类 ConcreteCommand1ConcreteCommand2,它们都继承自 Command 接口,并包含一个指向 Receiver 对象的指针。在 execute 方法中,它们会调用接收者对象的相应方法来执行具体操作。

接着,我们定义了一个宏命令类 MacroCommand,它由多个命令组成,通过 addCommand 方法可以添加命令。在 execute 方法中,它会依次执行每个命令。

然后,我们定义了一个命令发起者 Invoker,它包含一个成员变量 m_cmd,指向一个命令对象,通过 setCommand 方法设置命令对象,通过 executeCommand 方法来执行命令。

最后,我们在客户端代码中创建一个接收者对象和两个具体命令对象,然后将它们添加到一个宏命令对象中。将宏命令对象设置为命令发起者的命令,并执行命令。

这个命令模式实现还包括了宏命令,宏命令是由多个命令组成的一个命令,它可以一次性执行多个命令。在示例中,我们定义了一个 MacroCommand 类来实现宏命令,它包含了一个 std::vector 来存储多个命令对象。在 execute 方法中,它会依次执行每个命令。

命令模式的优点是可以将命令的请求者和命令的执行者解耦,使得请求者和执行者可以独立变化。同时,命令模式也支持撤销和重做操作,只需保存每个命令的执行历史记录即可实现。

需要注意的是,命令模式的实现并不一定要包含接收者和宏命令,这只是一个示例。在实际应用中,可以根据需要进行设计。

  • C++函数对象
    Command模式与C++中的函数对象有些类似。但两者定义行为接口的规范有所区别: Command以面向对象中的“接口-实现”来定义行为接口规范,更严格,但有性能损失;C++函数对象以函数签名来定义行为接口规范,更灵活,性能更高。

相关文章:

《设计模式》命令模式

《设计模式》命令模式 命令模式&#xff08;Command Pattern&#xff09;是一种行为型设计模式&#xff0c;它将请求和处理分开&#xff0c;使得请求发送者和接收者解耦&#xff0c;从而降低系统的耦合度。在命令模式中&#xff0c;请求被封装为一个独立的对象&#xff0c;并且…...

开源物联网平台有哪些?

目前市面上有许多开源物联网平台可供选择。以下是其中一些较为流行和知名的平台&#xff1a; Eclipse IoT&#xff1a;Eclipse IoT 是一个开源的物联网平台&#xff0c;旨在提供可扩展、灵活和高度集成的工具和框架&#xff0c;用于构建、部署和管理 IoT 解决方案。它包含多个…...

Tesla Autopilot,处理器和硬件

作者 | 初光 出品 | 车端 备注 | 转载请阅读文中版权声明 知圈 | 进“汽车电子与AutoSAR开发”群&#xff0c;请加微“cloud2sunshine” 总目录链接>> AutoSAR入门和实战系列总目录 Tesla MOdelS/X 中有 60 多个处理器。其他型号的处理器较少&#xff0c;但数量仍然不少…...

jianzhiOffer第二版难重点记录

04. 二维数组中的查找https://leetcode.cn/problems/er-wei-shu-zu-zhong-de-cha-zhao-lcof/ 思路&#xff1a;可以每层用以恶搞二分查找&#xff0c;优化思路&#xff1a;从左下角出发直接用二分。 ​​​​​​07. 重建二叉树https://leetcode.cn/problems/zhong-jian-er-cha…...

C语言 | 问题20230225

C语言 | 问题20230225 文章目录C语言 | 问题202302251.问题1无限循环2.问题2C 中的运算符优先级实例1&#xff1a;1.问题1 Which slice of the following code is NOT endless loop? 以下代码的哪一部分不是无限循环&#xff1f; A for (;(cgetchar())!\n; ) printf("*c&…...

【机器学习笔记】Python基础笔记

目录基础语法加载数据&#xff1a;pd.read_csv查看数据大小&#xff1a;shape浏览数据行字段&#xff1a;columns浏览少量数据&#xff1a;head()浏览数据概要&#xff1a;describe()基础功能语法缺省值去除缺失值&#xff1a;dropna按行删除&#xff1a;存在空值&#xff0c;即…...

js-DOM03-DOM对CSS的操作

DOM对CSS的操作 - 读取和修改内联样式 - 使用style属性来操作元素的内联样式 - 读取内联样式&#xff1a; 语法&#xff1a;元素.style.样式名 - 例子&#xff1a; 元素.style.width 元素.style.…...

tun驱动之tun_init

tun驱动的初始化方法是tun_init。 static int __init tun_init(void) {int ret 0;pr_info("%s, %s\n", DRV_DESCRIPTION, DRV_VERSION);ret rtnl_link_register(&tun_link_ops);if (ret) {pr_err("Cant register link_ops\n");goto err_linkops;}re…...

模拟退火算法优化bp

%% 基于模拟退火遗传算法优化BP神经网络的钢带厚度预测 clear clc close all format short %% 加载训练数据 Xtrxlsread(train_data.xlsx); DDsize(Xtr,2); input_trainXtr(:,1:DD-1);% output_trainXtr(:,DD);% %% 加载测试数据 Xtexlsread(test_data.xlsx); input_testXte(…...

【NFC音乐相册】简易制作

欢迎来到 Claffic 的博客 &#x1f49e;&#x1f49e;&#x1f49e; 前言&#xff1a; NFC音乐相册在前段时间火了一把&#xff0c;想必大家都听过了&#xff0c;最近我刷到了这个东西&#xff0c;闲来无事就弄了几个&#xff0c;这篇博客就记录下制作工序。 注&#xff1a;我所…...

每日一题——L1-085 试试手气(15)

L1-085 试试手气 我们知道一个骰子有 6 个面&#xff0c;分别刻了 1 到 6 个点。下面给你 6 个骰子的初始状态&#xff0c;即它们朝上一面的点数&#xff0c;让你一把抓起摇出另一套结果。假设你摇骰子的手段特别精妙&#xff0c;每次摇出的结果都满足以下两个条件&#xff1a;…...

FreeRTOS信号量

前面介绍过&#xff0c;队列&#xff08;queue&#xff09;可以用于传输数据&#xff1a;在任务之间&#xff0c;任务和中断之间。消息队列用于传输多个数据&#xff0c;但是有时候我们只需要传递一个状态&#xff0c;这个状态值需要用一个数值表示&#xff0c;比如&#xff1a…...

Leetcode.2385 感染二叉树需要的总时间

题目链接 Leetcode.2385 感染二叉树需要的总时间 Rating &#xff1a; 1711 题目描述 给你一棵二叉树的根节点 root&#xff0c;二叉树中节点的值 互不相同 。另给你一个整数 start。在第 0分钟&#xff0c;感染 将会从值为 start的节点开始爆发。 每分钟&#xff0c;如果节点…...

[蓝桥杯 2022 国 B] 卡牌(贪心/二分)

题目传送门 该题第一思路是想去模拟题目中所描述的过程 这里我选择从大到小遍历可能凑出的牌套数&#xff0c;计算凑出它需要补的牌数以及判断是否会超出能补的牌数 #include<iostream> #include<climits> #include<vector> #include<algorithm> #def…...

1301:大盗阿福

经典的dp打家劫舍问题状态设计dp[i][0]&#xff1a;在前i个店铺中选&#xff0c;且不选第i家的最大和dp[i][1]&#xff1a;在前i个店铺中选&#xff0c;且选第i家的最大和状态转移dp[i][0] max(dp[i-1][1], dp[i-1][0];第i家店不选&#xff0c;那么我们可以选第i-1个店 也可以…...

Netty——序列化的作用及自定义协议

序列化的作用及自定义协议序列化的重要性大小对比效率对比自定义协议序列化数据结构自定义编码器自定义解码器安全性验证NettyClientNettyServerNettyClientTestHandlerNettyServerTestHandler结果上一章已经说了怎么解决沾包和拆包的问题&#xff0c;但是这样离一个成熟的通信…...

一起Talk Android吧(第五百零五回:如何调整组件在约束布局中的大小)

文章目录 背景介绍调整方法各位看官们大家好,上一回中咱们说的例子是"如何调整组件在约束布局中的位置",这一回中咱们说的例子是" 如何调整组件在约束布局中的大小"。闲话休提,言归正转, 让我们一起Talk Android吧! 背景介绍 在使用约束(constraintl…...

【数据库】数据库的完整性

第五章 数据库完整性 数据库完整性 数据库的完整性是指数据的正确性和相容性 数据的正确性是指数据是符合现实世界语义&#xff0c;反映当前实际状况的数据的相容性是指数据库的同一对象在不同的关系中的数据是符合逻辑的 关系模型中有三类完整性约束&#xff1a;实体完整性…...

基因净化车间装修设计方案SICOLAB

基因净化车间的设计方案应该根据实际需求进行定制&#xff0c;以下是一些规划建设要点和洁净设计要注意的事项&#xff1a;一、净化车间规划建设要点&#xff1a;&#xff08;1&#xff09;基因车间的面积应该根据实验项目的规模进行规划&#xff0c;包括充足的操作区域和足够的…...

java 内部类的四种“写法”

基本介绍语法格式分类成员内部类静态内部类局部内部类匿名内部类&#xff08;&#x1f402;&#x1f58a;&#xff09;一、基本介绍 : 1.概述当一个类的内部又完整地嵌套了另一个类时&#xff0c;被嵌套于内部的“内核”我们称之为“内部类”(inner class)&#xff1b;而包含该…...

手游刚开服就被攻击怎么办?如何防御DDoS?

开服初期是手游最脆弱的阶段&#xff0c;极易成为DDoS攻击的目标。一旦遭遇攻击&#xff0c;可能导致服务器瘫痪、玩家流失&#xff0c;甚至造成巨大经济损失。本文为开发者提供一套简洁有效的应急与防御方案&#xff0c;帮助快速应对并构建长期防护体系。 一、遭遇攻击的紧急应…...

SkyWalking 10.2.0 SWCK 配置过程

SkyWalking 10.2.0 & SWCK 配置过程 skywalking oap-server & ui 使用Docker安装在K8S集群以外&#xff0c;K8S集群中的微服务使用initContainer按命名空间将skywalking-java-agent注入到业务容器中。 SWCK有整套的解决方案&#xff0c;全安装在K8S群集中。 具体可参…...

如何在看板中体现优先级变化

在看板中有效体现优先级变化的关键措施包括&#xff1a;采用颜色或标签标识优先级、设置任务排序规则、使用独立的优先级列或泳道、结合自动化规则同步优先级变化、建立定期的优先级审查流程。其中&#xff0c;设置任务排序规则尤其重要&#xff0c;因为它让看板视觉上直观地体…...

【大模型RAG】Docker 一键部署 Milvus 完整攻略

本文概要 Milvus 2.5 Stand-alone 版可通过 Docker 在几分钟内完成安装&#xff1b;只需暴露 19530&#xff08;gRPC&#xff09;与 9091&#xff08;HTTP/WebUI&#xff09;两个端口&#xff0c;即可让本地电脑通过 PyMilvus 或浏览器访问远程 Linux 服务器上的 Milvus。下面…...

今日科技热点速览

&#x1f525; 今日科技热点速览 &#x1f3ae; 任天堂Switch 2 正式发售 任天堂新一代游戏主机 Switch 2 今日正式上线发售&#xff0c;主打更强图形性能与沉浸式体验&#xff0c;支持多模态交互&#xff0c;受到全球玩家热捧 。 &#x1f916; 人工智能持续突破 DeepSeek-R1&…...

select、poll、epoll 与 Reactor 模式

在高并发网络编程领域&#xff0c;高效处理大量连接和 I/O 事件是系统性能的关键。select、poll、epoll 作为 I/O 多路复用技术的代表&#xff0c;以及基于它们实现的 Reactor 模式&#xff0c;为开发者提供了强大的工具。本文将深入探讨这些技术的底层原理、优缺点。​ 一、I…...

GruntJS-前端自动化任务运行器从入门到实战

Grunt 完全指南&#xff1a;从入门到实战 一、Grunt 是什么&#xff1f; Grunt是一个基于 Node.js 的前端自动化任务运行器&#xff0c;主要用于自动化执行项目开发中重复性高的任务&#xff0c;例如文件压缩、代码编译、语法检查、单元测试、文件合并等。通过配置简洁的任务…...

MFC 抛体运动模拟:常见问题解决与界面美化

在 MFC 中开发抛体运动模拟程序时,我们常遇到 轨迹残留、无效刷新、视觉单调、物理逻辑瑕疵 等问题。本文将针对这些痛点,详细解析原因并提供解决方案,同时兼顾界面美化,让模拟效果更专业、更高效。 问题一:历史轨迹与小球残影残留 现象 小球运动后,历史位置的 “残影”…...

打手机检测算法AI智能分析网关V4守护公共/工业/医疗等多场景安全应用

一、方案背景​ 在现代生产与生活场景中&#xff0c;如工厂高危作业区、医院手术室、公共场景等&#xff0c;人员违规打手机的行为潜藏着巨大风险。传统依靠人工巡查的监管方式&#xff0c;存在效率低、覆盖面不足、判断主观性强等问题&#xff0c;难以满足对人员打手机行为精…...

(一)单例模式

一、前言 单例模式属于六大创建型模式,即在软件设计过程中,主要关注创建对象的结果,并不关心创建对象的过程及细节。创建型设计模式将类对象的实例化过程进行抽象化接口设计,从而隐藏了类对象的实例是如何被创建的,封装了软件系统使用的具体对象类型。 六大创建型模式包括…...