新闻中心
C++如何实现命令模式(Command Pattern)_C++设计模式与命令模式实现
命令模式通过封装请求为对象实现调用者与接收者解耦,支持撤销、重做和任务队列;其核心角色包括Command接口、ConcreteCommand、Receiver、Invoker和Client;示例中以TextEditor为接收者,OpenCommand和S*eCommand为具体命令,Button为调用者,通过execute()执行操作;扩展undo()方法可实现撤销功能,CommandHistory记录命令历史;C++11后可用std::function和lambda简化实现;命令应保持独立无状态以避免副作用。

命令模式是一种行为设计模式,它将请求封装为对象,从而使你可以用不同的请求、队列、日志来参数化其他对象。在C++中,命令模式通过抽象命令接口、具体命令类和接收者(Receiver)协作实现解耦。这种模式常用于实现撤销/重做、任务队列、远程调用等场景。
命令模式的核心结构
命令模式通常包含以下几个角色:
- Command(命令接口):声明执行操作的接口,通常是一个虚函数execute()。
- ConcreteCommand(具体命令):实现Command接口,持有一个接收者对象,并在execute()中调用接收者的相应方法。
- Receiver(接收者):真正执行请求的对象,包含具体的业务逻辑。
- Invoker(调用者):持有命令对象,通过调用命令的execute()来触发请求。
- Client(客户端):创建命令对象并绑定接收者,然后将命令交给调用者。
基础实现示例
以下是一个简单的文本编辑器中“打开文件”和“保存文件”命令的实现:
#include <iostream>
#include <memory>
#include <vector>
// 接收者:执行实际操作
class TextEditor {
public:
void open() {
std::cout << "TextEditor: 打开文件\n";
}
void s*e() {
std::cout << "TextEditor: 保存文件\n";
}
};
// 命令接口
class Command {
public:
virtual ~Command() = default;
virtual void execute() = 0;
};
// 具体命令:打开文件
class OpenCommand : public Command {
private:
TextEditor& editor;
public:
OpenCommand(TextEditor& e) : editor(e) {}
void execute() override {
editor.open();
}
};
// 具体命令:保存文件
class S*eCommand : public Command {
private:
TextEditor& editor;
public:
S*eCommand(TextEditor& e) : editor(e) {}
void execute() override {
editor.s*e();
}
};
// 调用者:按钮或快捷键
class Button {
private:
std::unique_ptr<Command> command;
public:
void setCommand(std::unique_ptr<Command> cmd) {
command = std::move(cmd);
}
void click() {
if (command) {
command->execute();
}
}
};使用方式如下:
int main() {
TextEditor editor;
// 客户端创建命令并绑定接收者
auto openCmd = std::make_unique<OpenCommand>(editor);
auto s*eCmd = std::make_unique<S*eCommand>(editor);
// 设置到调用者(如按钮)
Button openButton, s*eButton;
openButton.setCommand(std::move(openCmd));
s*eButton.setCommand(std::move(s*eCmd));
// 模拟用户点击
openButton.click(); // 输出:TextEditor: 打开文件
s*eButton.click(); // 输出:TextEditor: 保存文件
return 0;
}支持撤销操作的扩展
命令模式天然适合实现撤销功能。只需在命令接口中增加undo()方法:
易标AI
告别低效手工,迎接AI标书新时代!3分钟智能生成,行业唯一具备查重功能,自动避雷废标项
135
查看详情
class Command {
public:
virtual ~Command() = default;
virtual void execute() = 0;
virtual void undo() {} // 可选,默认为空
};
class S*eCommand : public Command {
private:
TextEditor& editor;
public:
S*eCommand(TextEditor& e) : editor(e) {}
void execute() override {
editor.s*e();
}
void undo() override {
std::cout << "TextEditor: 撤销保存\n";
}
};调用者可以记录历史命令以支持撤销:
class CommandHistory {
private:
std::vector<std::unique_ptr<Command>> history;
public:
void push(std::unique_ptr<Command> cmd) {
history.push_back(std::move(cmd));
}
void undo() {
if (!history.empty()) {
history.back()->undo();
history.pop_back();
}
}
};使用函数对象简化实现
C++11以后,可以用std::function和lambda表达式简化命令模式:
#include <functional>
class FunctionalCommand : public Command {
private:
std::function<void()> doAction;
std::function<void()> undoAction;
public:
FunctionalCommand(std::function<void()> exec, std::function<void()> un = []{})
: doAction(exec), undoAction(un) {}
void execute() override {
doAction();
}
void undo() override {
undoAction();
}
};
// 使用示例
TextEditor editor;
auto cmd = std::make_unique<FunctionalCommand>(
[&editor]() { editor.s*e(); },
[]() { std::cout << "撤销保存\n"; }
);基本上就这些。命令模式通过将操作封装成对象,实现了调用者与接收者的解耦,增强了系统的可扩展性和灵活性。不复杂但容易忽略的是,每个命令应尽量保持独立和无状态,避免共享数据带来的副作用。
以上就是C++如何实现命令模式(Command Pattern)_C++设计模式与命令模式实现的详细内容,更多请关注其它相关文章!
# 客户端
# 广西网络营销推广方式
# 湖南网站优化电脑
# 全员营销推广奖励方案
# 郴州网站建设现状
# 诸城关键词搜索排名
# 宁夏seo网络营销广告
# 嘉定区抖音营销推广招聘
# 专业网站建设在哪里买书
# 中关村企业网站建设
# 祭司素材网站建设需要
# 重做
# ai
# 的是
# 绑定
# 可以用
# 如何实现
# 保存文件
# 是一个
# 游戏开发
# 调用者
# stream
# ios
# c++
相关栏目:
【
科技资讯46185 】
【
网络学院92790 】
相关推荐:
CSS实现侧边栏导航项全宽圆角悬停背景效果
win11怎么查看应用耗电情况 Win11电池设置查看应用能耗排行榜【优化】
Win11 BitLocker密码忘了怎么办 Win11找回BitLocker恢复密钥方法【解决】
Go语言HTML解析:利用Goquery精准获取指定元素内容
mysql如何设置表访问权限_mysql表访问权限配置
微信商城在哪里打开【步骤】
在Typer应用中优雅地处理和重组任意命令行参数
“音游” × “怪文书” 题材的节奏冒险游戏 《晕晕电波症候群》确定于2026年4月发售!
从OpenAI API响应中高效提取生成文本
2026春节假期时间安排 2026春节假日查询
解决macOS上安装pyhdf时‘hdf.h’文件缺失的编译错误
C++的std::forward_list怎么用_C++ STL中单向链表容器的特点与应用
PDO预处理语句中冒号的正确处理:区分SQL函数格式与命名占位符
J*aScript生成器_j*ascript异步迭代
解决 Vaadin 8 中大文件音频播放与定位时出现的 IOException
漫画星球免费下拉式入口 漫画星球免费漫画在线阅读网站
Golang切片为何属于引用类型_Golang slice底层结构与引用语义说明
Golang如何通过reflect操作map_Golang reflect map操作与遍历技巧
html怎么运行外部js文件中的函数_运html外js文件函数法【技巧】
QQ邮箱电脑版登录入口_QQ邮箱官方网站登录平台
神庙逃亡小游戏在线玩 神庙逃亡小游戏入口
CSS Flexbox与媒体查询:实现响应式布局中元素的并排与堆叠
Win11怎么用U盘重装系统 Win11制作启动盘并重装系统完整教程【详解】
迅雷下载到U盘速度很慢怎么办_迅雷U盘下载慢优化方法
Python实时数据流中的动态最值查找策略
mc.js游戏直达 mc.js网页免下载版本秒进地址
sublime怎么设置启动时打开的窗口_sublime会话管理与热退出
狙击外星人小游戏开始_狙击外星人小游戏立即开始
Web Components中自定义开关组件状态同步的常见陷阱与解决方案
Composer如何解决json扩展缺失的错误
excel如何生成目录 excel一键生成工作表目录超链接
海棠电脑版入口_通过电脑访问海棠官网阅读
自定义Bag-of-Words实现:处理带负号的词汇权重
Yandex官方入口网址 Yandex俄罗斯搜索引擎最新在线地址
谷歌浏览器最新官方入口链接 谷歌浏览器网页版官网导航
Win10磁盘清理工具在哪 Win10打开并使用磁盘清理【教程】
J*aScript 字符串标签转换:使用正则表达式高效替换
qq游戏免费畅玩入口_qq游戏电脑版快速启动
海量存储:机器视觉智能化的核心基石
Bing引擎入口最新2025 Bing搜索免费官方登录
QQ邮箱登录官网首页 腾讯QQ邮箱网页入口
怎么在mac上运行html代码_mac运行html代码方法【指南】
CSS Grid如何控制元素对齐_align-items与justify-items组合使用
生成rdflib自定义SPARQL函数:参数匹配与实践指南
GemBox Document HTML转PDF垂直文本渲染问题及解决方案
192.168.1.1管理中心入口 192.168.1.1路由器网页设置平台
想当下一个《2077》?《心之眼》Steam评价升至"多半好评"
Python实现多节点属性重叠度分析教程
《燕云十六声》两周内达九百万玩家!位居畅销榜第五
Golang指针如何与map组合使用_Golang map指针组合实践


2025-11-08
浏览次数:次
返回列表
std::cout << "TextEditor: 保存文件\n";
}
};
// 命令接口
class Command {
public:
virtual ~Command() = default;
virtual void execute() = 0;
};
// 具体命令:打开文件
class OpenCommand : public Command {
private:
TextEditor& editor;
public:
OpenCommand(TextEditor& e) : editor(e) {}
void execute() override {
editor.open();
}
};
// 具体命令:保存文件
class S*eCommand : public Command {
private:
TextEditor& editor;
public:
S*eCommand(TextEditor& e) : editor(e) {}
void execute() override {
editor.s*e();
}
};
// 调用者:按钮或快捷键
class Button {
private:
std::unique_ptr<Command> command;
public:
void setCommand(std::unique_ptr<Command> cmd) {
command = std::move(cmd);
}
void click() {
if (command) {
command->execute();
}
}
};