新闻中心
C++如何实现一个命令模式_C++设计模式之请求封装与撤销/重做功能
命令模式将请求封装为对象,支持参数化操作与撤销重做。通过Command接口、具体命令类、接收者、调用者和客户端协作,实现文本编辑器的插入删除及历史控制,便于扩展与维护。

在C++中实现命令模式,核心是将“请求”封装成独立对象,使得可以用不同的请求对客户进行参数化,并支持请求的撤销、重做、排队等操作。这种设计特别适用于需要支持操作历史记录的场景,比如文本编辑器、图形界面操作或游戏中的技能系统。
命令模式的基本结构
命令模式包含以下几个关键角色:
- Command(命令接口):声明执行操作的接口,通常包含一个execute()方法,有时还包括undo()和redo()。
- ConcreteCommand(具体命令):实现Command接口,持有对Receiver的引用,并在execute()中调用Receiver的相应方法。
- Receiver(接收者):真正执行请求的对象,比如文档、画布或游戏角色。
- Invoker(调用者):持有命令对象,通过调用命令的execute()来触发请求,例如按钮或菜单项。
- Client(客户端):创建ConcreteCommand对象,并设置其接收者。
举个例子:一个简单的文本编辑器支持“插入文本”和“删除文本”操作,并能撤销上一步。
基础实现:支持执行与撤销
先定义命令基类:
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++设计模式之请求封装与撤销/重做功能的详细内容,更多请关注其它相关文章!
# 象中
# 网站建设的条件分析
# 旅游网站建设开发费用
# 烘培店品牌网站推广
# 淘客客网站的推广渠道
# 昆明模板网站推广方案
# 大众点评如何推广营销
# 桥西区个人网站优化记录
# 木材板材网站定制建设
# 厦门昆山网站建设
# 临沂做网站建设公司
# 几个
# 迭代
# c++
# 如何处理
# 客户端
# 如何实现
# 调用者
# 编辑器
# 如何使用
# 重做
# red
# ai
# 栈
# 浏览器
# 命令模式
相关栏目:
【
科技资讯46185 】
【
网络学院92790 】
相关推荐:
蛙漫官网漫画入口地址_蛙漫在线畅读无广告弹窗
HTML空白字符处理机制:渲染、DOM与编码实践
印象笔记如何设提醒任务防漏执行_印象笔记设提醒任务防漏执行【任务提醒】
使用 Pandas 高效处理 .dat 文件:字符清理与数据计算
TikTok搜索结果不显示如何解决 TikTok搜索刷新优化方法
J*aScript中安全有效地处理localStorage字符串数据
Pygame教程:解决用户输入与游戏状态更新不同步问题
CSS Flexbox与媒体查询:实现响应式布局中元素的并排与堆叠
在J*a项目里如何构建对象之间的契约_接口约束的实际落地
在J*a里如何理解依赖关系的方向_依赖方向在模块结构中的作用
Win11 USB传输速度慢怎么解决 Win11 USB驱动更新与设置
C++ map遍历方法大全_C++ map迭代器使用总结
C++编译期如何执行复杂计算_C++模板元编程(TMP)技巧与应用
天猫2025双十一0点秒杀攻略 天猫爆款抢购时间
铁路12306改签能改到更早的车次吗_铁路12306改签提前车次规则
荣耀Play7TPro怎样在信息App置顶客服对话_iPhone荣耀Play7TPro信息App置顶客服对话【优先查看】
2026春节假期时间安排 2026春节假日查询
红果短剧网页版官网入口 官方最新网址发布
Win11怎么开启省电模式_Win11电池节电模式自动开启
响应式容器内容自动缩放与宽高比维持教程
sublime怎么设置启动时打开的窗口_sublime会话管理与热退出
AO3最新镜像入口 Archive of Our Own官方平台访问
在Socket.IO连接中实现Access Token自动更新与动态重连
必由学官方平台入口 必由学在线课堂登录地址
css卡片内容溢出如何处理_使用overflow隐藏或scroll显示内容
如何在J*a中使用Locale处理多语言环境
React Hooks最佳实践:动态组件状态管理的组件化方案
漫蛙漫画官方主页入口 漫蛙MANWA网页直达访问链接
steam官方入口大全 steam账号注册及操作指南
晋江读书网页版在线登录 晋江读书电脑版官网
快手网页版在线登录 快手网页版官网入口快速访问
深入理解Google Cloud Datastore查询:祖先路径与数据一致性
Fabric Mod开发:在1.19.3+版本中正确添加自定义物品并管理物品组
虫虫漫画精品漫画官网_虫虫漫画精品漫画官网进入精品漫画
离线运行Go语言之旅:本地部署与GOPATH配置指南
Golang如何使用new_Go new分配内存机制讲解
PDF文件体积过大处理_PDF压缩技巧详解
b站怎么取消点赞_b站点赞取消操作方法
vivo浏览器自带的下载器速度慢怎么办 vivo浏览器提升文件下载速度的技巧
如何修改开机登录密码_Windows账户安全设置超详细教程【必学】
win11如何加载ICC颜色配置文件 Win11校色文件安装与显示器色彩管理【指南】
葱吃多了会怎样 葱吃多了会伤胃吗
微信客户端如何收红包_微信客户端接收红包使用教程
Lar*el如何生成PDF或Excel文件_Lar*el文档导出工具与使用教程
wps文字怎么插入目录并自动更新_wps文字如何插入目录并自动更新方法
Win10系统服务哪些可以禁用 Win10安全优化服务列表【干货】
响应式CSS Grid布局:优化网格项在小屏幕下的堆叠与宽度适配
AO3最新入口2025公告_AO3中文官网合集
qq游戏手机版下载安装_qq游戏移动端入口
微信语音通话掉线如何解决 微信语音通话稳定优化方法


2025-12-02
浏览次数:次
返回列表
e class="brush:php;toolbar:false;">void execute() override {
editor->insert(text);
}
void undo() override {
editor->erase(text.size());
}