命令模式将请求封装为对象,支持参数化操作与撤销重做。通过Command接口、具体命令类、接收者、调用者和客户端协作,实现文本编辑器的插入删除及历史控制,便于扩展与维护。

在C++中实现命令模式,核心是将“请求”封装成独立对象,使得可以用不同的请求对客户进行参数化,并支持请求的撤销、重做、排队等操作。这种设计特别适用于需要支持操作历史记录的场景,比如文本编辑器、图形界面操作或游戏中的技能系统。
命令模式包含以下几个关键角色:
举个例子:一个简单的文本编辑器支持“插入文本”和“删除文本”操作,并能撤销上一步。
先定义命令基类:
class Command {
public:
virtual ~Command() = default;
virtual void execute() = 0;
virtual void undo() = 0;
};
定义接收者——文本编辑器:
class TextEditor {
std::string content;
public:
void insert(const std::string& text) {
content += text;
}
void erase(size_t len) {
if (len >= content.size()) {
content.clear();
} else {
content.erase(content.size() - len);
}
}
std::string getContent() const { return content; }
};
实现具体命令:
Machine Translation
聚合多个来源的AI翻译
49
查看详情
class InsertCommand : public Command {
TextEditor* editor;
std::string text;
public:
InsertCommand(TextEditor* e, const std::string& t) : editor(e), text(t) {}
<pr
e class="brush:php;toolbar:false;">void execute() override {
editor->insert(text);
}
void undo() override {
editor->erase(text.size());
}};
class DeleteCommand : public Command { TextEditor editor; std::string deletedText; public: DeleteCommand(TextEditor e, size_t len) : editor(e) { // 假设我们知道要删多少,实际中可由editor提供 deletedText = editor->getContent().substr(editor->getContent().size() - len); }
void execute() override {
editor->erase(deletedText.size());
}
void undo() override {
editor->insert(deletedText);
}};
引入一个调用者类来保存命令序列,支持撤销和重做:
class CommandHistory {
std::vector<std::unique_ptr<Command>> commands;
int current = -1; // 当前位置,用于支持redo
<p>public:
void execute(std::unique_ptr<Command> cmd) {
cmd->execute();
// 清除当前位置之后的历史(类似浏览器行为)
if (current < (int)commands.size() - 1) {
commands.erase(commands.begin() + current + 1, commands.end());
}
commands.push_back(std::move(cmd));
current++;
}</p><pre class="brush:php;toolbar:false;">void undo() {
if (current >= 0) {
commands[current]->undo();
current--;
}
}
void redo() {
if (current < (int)commands.size() - 1) {
commands[current + 1]->execute();
current++;
}
}};
客户端代码演示如何组合这些部分:
int main() {
TextEditor editor;
CommandHistory history;
<pre class="brush:php;toolbar:false;">auto insertHello = std::make_unique<InsertCommand>(&editor, "Hello");
auto insertWorld = std::make_unique<InsertCommand>(&editor, " World");
history.execute(std::move(insertHello)); // Hello
history.execute(std::move(insertWorld)); // Hello World
std::cout << editor.getContent() << "\n"; // 输出: Hello World
history.undo(); // 撤销" World"
std::cout << editor.getContent() << "\n"; // 输出: Hello
history.redo(); // 重做
std::cout << editor.getContent() << "\n"; // 输出: Hello World
return 0;}
基本上就这些。通过命令模式,你把“动作”变成了可存储、可传递的对象。它让调用者不依赖具体操作,也方便扩展新命令而不改动现有代码。配合历史栈,轻松实现撤销/重做功能。对于更复杂的场景,可以加入命令合并(如连续输入合并为一次)、命令序列(宏)、持久化等特性。
以上就是C++如何实现一个命令模式_C++设计模式之请求封装与撤销/重做功能的详细内容,更多请关注其它相关文章!
相关文章:
J*aScript中安全有效地处理localStorage字符串数据
Python Sounddevice 音频卡顿问题解析与队列数据安全处理
Win11怎么查看电脑配置_Win11硬件配置检测工具使用
优化Lar*el Docker镜像:Composer与PHP版本控制策略
mysql如何分析事务日志_mysql事务日志分析方法
抖音网页版平台入口 抖音网页版官网在线访问教程
怎样把文件彻底粉碎无法恢复_Windows下安全删除敏感数据【隐私保护】
PHP表单提交消息延迟显示:Post-Redirect-Get模式深度解析与实践
一加手机电池耗电快怎么办_一加手机电池耗电快的解决方法
Win11输入法不见了怎么办_Windows11恢复语言栏显示方法
Go语言中Map值调用指针接收器方法的限制与应对
Android Studio计算器C键逻辑错误排查与修复:条件判断优化指南
在J*a中如何开发简易电子商务商品管理系统_商品管理系统项目实战解析
如何在复杂的电商平台中优雅地管理共享资源并确保正确重定向,使用spryker-shop/resource-share-page模块助你一臂之力
Golang如何实现Web接口签名验证_Golang Web接口签名校验开发方法
PHP表单数据传递:如何通过隐藏输入字段获取动态ID
QQ邮箱稳定登录入口_QQ邮箱官方网站网页版使用
俄罗斯方块最新版入口 俄罗斯方块在线玩官网入口
谷歌浏览器最新官方入口链接 谷歌浏览器网页版官网导航
格力空气能E5故障代码是什么情况_格力空气能E5代码解析与应对措施
Golang如何优雅处理error_Golang error处理最佳实践总结
C++如何使用AddressSanitizer(ASan)_C++调试工具中检测内存访问错误的利器
Excel中VLOOKUP的第四个参数是干什么用的_Excel VLOOKUP第四参数作用解析
C#如何安全地从用户上传的XML文件中读取数据? 验证与清理策略
顺丰快递查单号物流信息 顺丰快递小程序查询入口
AO3镜像入口大全 AO3网页版内容访问全集
Android Studio计算器C键功能异常排查与修复教程
JUnit5/Mockito:优雅测试内部依赖与异常处理的实践
PHP面向对象编程中避免重复创建PDO数据库连接的最佳实践
Go RPC HTTP服务正确实现与常见陷阱解析
微信网页版扫码登录入口 微信网页版二维码登录入口
Python异步编程实践:使用Binance API构建实时交易数据流
126邮箱手机版登录官网2026_126手机邮箱免费入口最新
J*aScript Promise链中如何正确终止后续.then执行并处理错误
我的世界mc.js免费游戏直接能玩 我的世界mc.js小游戏免费秒玩入口
Go语言中动态执行代码字符串的策略与实践
蛙漫画网页版全站入口 蛙漫热门作品免费浏览
Spring Boot嵌入式服务器与J*a EE:功能支持深度解析
蛙漫2台版漫画地址 Manwa2正版网页版链接
Sublime Text怎么设置垂直标尺_Sublime配置Rulers规范代码长度
动漫花园资源网使用步骤_动漫花园资源网下载流程
Yandex搜索引擎官网入口_俄罗斯Yandex免登录一键直达
从J*aScript对象中精确提取指定属性的教程
mysql通配符支持数字匹配吗_mysql通配符能否用于数字匹配的解析
mc.js游戏直达 mc.js网页免下载版本秒进地址
React Router 嵌套组件中 URL 重定向问题的解决方案
在J*a里如何理解依赖关系的方向_依赖方向在模块结构中的作用
必由学官方登录入口 必由学教师学生账号快速访问
优化Django表单:提交验证失败后保留用户输入
css绝对定位元素脱离父容器怎么办_确保父元素position非static