news 2026/4/16 5:36:19

设计模式[14]——命令模式一分钟彻底说透

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
设计模式[14]——命令模式一分钟彻底说透

设计模式[14]——命令模式(Command)一分钟彻底说透(C++版·软件领域真实例子)

一句话定义

将“请求”(操作)封装成一个独立的对象,从而让发送者执行者完全解耦,支持撤销、重做、队列、日志等高级功能。

最狠的比喻(软件人专属)

图形编辑器(如Photoshop/GIMP)的“撤销/重做”:

  • 每按一次“调亮度”“画笔”“裁剪”都生成一个命令对象
  • 放进历史栈 → 按Ctrl+Z就弹出上一个命令执行undo()
  • 客户端(菜单/快捷键)只知道“执行命令”,完全不知道具体怎么调亮度
为什么需要它?(坏味道瞬间爆炸)

不用命令模式,你会直接这样写:

editor.brighten();// 菜单直接调用编辑器方法editor.crop();// 想加撤销?每个操作都要在编辑器里写undo逻辑,耦合到死// 想队列批量执行?想记录操作日志?想宏命令?全寄!
和之前模式彻底分清(10秒表)
项目责任链(Chain of Responsibility)装饰器(Decorator)命令(Command)
核心意图请求沿链找处理者动态叠加行为将操作封装成对象,支持撤销/队列
关键元素next指针 + 可中断包装链 + 全执行execute() + undo()
执行时机立即沿链传递立即层层执行可延迟、可队列、可撤销
典型场景Web中间件、事件过滤流加密/日志编辑器撤销、事务、宏命令、线程池
口号“传下去,直到有人接”“层层叠加”“命令即对象,想干啥都行”
真实软件例子:图形编辑器撤销/重做系统
#include<iostream>#include<memory>#include<vector>#include<string>usingnamespacestd;// 1. 接收者(真正干活的对象)classImageEditor{string state="原始图片";public:voidbrighten(){state+=" → 调亮";cout<<"[Editor] 执行调亮 → 当前状态: "<<state<<endl;}voidcrop(){state+=" → 裁剪";cout<<"[Editor] 执行裁剪 → 当前状态: "<<state<<endl;}voidunbrighten(){/* 实际项目里恢复状态 */cout<<"[Editor] 撤销调亮\n";}voiduncrop(){cout<<"[Editor] 撤销裁剪\n";}voidshow()const{cout<<"当前图片状态: "<<state<<endl;}};// 2. 命令接口classCommand{public:virtual~Command()=default;virtualvoidexecute()=0;virtualvoidundo()=0;};// 3. 具体命令(绑定接收者 + 操作)classBrightenCommand:publicCommand{ImageEditor*editor;public:explicitBrightenCommand(ImageEditor*e):editor(e){}voidexecute()override{editor->brighten();}voidundo()override{editor->unbrighten();}};classCropCommand:publicCommand{ImageEditor*editor;public:explicitCropCommand(ImageEditor*e):editor(e){}voidexecute()override{editor->crop();}voidundo()override{editor->uncrop();}};// 4. 调用者(维护命令历史栈)classCommandHistory{vector<unique_ptr<Command>>history;ImageEditor&editor;public:explicitCommandHistory(ImageEditor&e):editor(e){}voidexecuteCommand(unique_ptr<Command>cmd){cmd->execute();history.push_back(move(cmd));// 入栈}voidundo(){if(!history.empty()){history.back()->undo();history.pop_back();}else{cout<<"无操作可撤销\n";}}voidshowState()const{editor.show();}};
客户端:撤销重做爽到飞起
intmain(){ImageEditor editor;CommandHistoryhistory(editor);// 用户操作history.executeCommand(make_unique<BrightenCommand>(&editor));history.executeCommand(make_unique<CropCommand>(&editor));history.executeCommand(make_unique<BrightenCommand>(&editor));history.showState();// 原始图片 → 调亮 → 裁剪 → 调亮cout<<"\n=== 用户按 Ctrl+Z 撤销 ===\n";history.undo();// 撤销最后一次调亮history.undo();// 撤销裁剪history.showState();// 回到只调亮一次的状态}

输出:

[Editor] 执行调亮 → 当前状态: 原始图片 → 调亮 [Editor] 执行裁剪 → 当前状态: 原始图片 → 调亮 → 裁剪 [Editor] 执行调亮 → 当前状态: 原始图片 → 调亮 → 裁剪 → 调亮 当前图片状态: 原始图片 → 调亮 → 裁剪 → 调亮 === 用户按 Ctrl+Z 撤销 === [Editor] 撤销调亮 [Editor] 撤销裁剪 当前图片状态: 原始图片 → 调亮
C++ 真实项目里无处不在
  • GUI框架:Qt的QUndoCommand(编辑器、场景编辑器)
  • 游戏编辑器:Unreal Editor的Transaction系统
  • 事务管理:数据库事务(Commit/Rollback)
  • 线程池/任务队列:把Command放进队列异步执行
  • 宏命令:组合多个Command成一个“超级命令”(如“保存项目” = 保存文件 + 清理缓存 + 记录日志)
终极口诀(编辑器开发者专属)

“操作封装成对象,撤销重做随便搞;
菜单快捷键解耦,历史栈里随便跳!”

刻在DNA里的一句话

当你需要“支持撤销/重做、延迟执行、队列操作、日志记录”等高级行为,且不想让调用者和具体操作耦合时,
立刻上命令模式——把“要做什么”封装成对象,想怎么玩都行!

现在,命令模式彻底说透了!
下一期是解释器模式(Interpreter)[15],虽然是GoF里最冷门的,但也有真实用途。

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/4/14 9:43:47

Langchain-Chatchat如何实现文档切片与向量化存储?技术细节曝光

Langchain-Chatchat 如何实现文档切片与向量化存储&#xff1f;技术细节深度解析 在企业智能化浪潮中&#xff0c;一个日益突出的矛盾正被越来越多开发者关注&#xff1a;通用大语言模型&#xff08;LLM&#xff09;虽然“见多识广”&#xff0c;却对企业的私有知识束手无策。你…

作者头像 李华
网站建设 2026/4/15 14:40:13

RocketMQ 介绍及适用场景

一、RocketMQ 简介RocketMQ 是阿里巴巴开源的分布式消息中间件&#xff0c;属于 Apache 顶级项目。它最初诞生于阿里巴巴集团&#xff0c;旨在解决大规模、高并发、低延迟下的消息传递需求。RocketMQ 使用 Java 语言开发&#xff0c;具有高可用、高性能、可扩展、强一致性等特点…

作者头像 李华
网站建设 2026/4/12 20:27:58

Agentic Frontend: 灵活的AI助手与聊天机器人构建平台

Agentic Frontend: 灵活的AI助手与聊天机器人构建平台 在当今快速发展的技术时代&#xff0c;AI助手和聊天机器人正在不断地改变我们的工作和生活方式。为了更好地满足这一需求&#xff0c;CopilotKit提供了一个强大的React UI和优雅的基础设施&#xff0c;让开发者能够轻松构…

作者头像 李华
网站建设 2026/4/12 20:20:35

别再只盯着网关超时:一次 SAP CRM Fiori 批量加产品卡死的真凶,竟然是用户参数 CRM_EVENT_TRACE

在做 SAP CRM 的 Fiori 应用性能排查时,很多人第一反应会去看 SAP Gateway、OData 调用、HANA SQL、甚至网络链路。这个思路没错,但有一类问题特别容易把人带进坑里:同一个应用、同一个操作、不同用户表现天差地别。你用自己的账号测起来飞快,测试同事一上手就超时,怎么看…

作者头像 李华
网站建设 2026/4/15 4:42:59

用 Doxygen 打通 SAP ABAP 源码文档与 UML:从包级扫描到一键生成站点

软件维护最怕的不是代码多,而是知识散。对很多企业而言,核心业务逻辑分布在 SAP ABAP 的类、接口、函数组、增强点、DDIC 对象、CDS 视图、网关服务实现里,真正的业务为什么这么写往往只存在于少数资深同事脑子里,或零碎地躺在 SE80 的短文本、SE61 文档、方法注释、数据元…

作者头像 李华
网站建设 2026/4/14 2:41:20

内网穿透的应用-废片秒变大片!IOPaint 让修图新手也能轻松上手

文章目录前言【视频教程】1.什么是IOPaint&#xff1f;2.本地部署IOPaint3.IOPaint简单实用4.公网远程访问本地IOPaint5.内网穿透工具安装6.配置公网地址7.使用固定公网地址远程访问总结IOPaint 的 AI 修图能力与 cpolar 的远程访问结合&#xff0c;让图像处理突破设备和网络限…

作者头像 李华