C++如何实现一个命令(Command)设计模式?(代码示例)

发布时间 - 2025-12-29 00:00:00    点击率:
Command模式将请求封装为对象,支持参数化、撤销、日志与队列;核心是统一命令接口,解耦调用者与接收者;通过具体命令类持有接收者并委托执行,可扩展宏命令与智能指针管理。

Command 模式把请求封装成对象,使你可以用不同的请求对客户进行参数化、支持撤销操作、排队或日志记录。核心是定义统一的命令接口,让调用者(Invoker)不依赖具体操作细节。

定义命令接口与具体命令类

先声明抽象基类 Command,包含执行(execute)和可选的撤销(undo)方法;再为每个实际操作实现具体子类,内部持有接收者(Receiver)引用并委托执行逻辑。

例如实现一个简单灯开关控制:

class Light {
public:
    void turnOn() { std::cout << "Light is ON\n"; }
    void turnOff() { std::cout << "Light is OFF\n"; }
};

class Command { public: virtual ~Command() = default; virtual void execute() = 0; virtual void undo() = 0; };

class LightOnCommand : public Command { Light& light; public: explicit LightOnCommand(Light& l) : light(l) {} void execute() override { light.turnOn(); } void undo() override { light.turnOff(); } };

class LightOffCommand : public Command { Light& light; public: explicit LightOffCommand(Light& l) : light(l) {} void execute() override { light.turnOff(); } void undo() override { light.turnOn(); } }

引入调用者(Invoker)管理命令

Invoker 不知道命令具体做什么,只负责触发 executeundo。它可保存最近执行的命令,用于撤销;也可支持命令队列、宏命令等扩展。

一个带撤销功能的简单 Invoker 示例:

class RemoteControl {
    Command* lastCommand = nullptr;
public:
    void setCommand(Command* cmd) {
        lastCommand = cmd;
    }
    void pressButton() {
        if (lastCommand) lastCommand->execute();
    }
    void pressUndo() {
        if (lastCommand) lastCommand->undo();
    }
};

组合使用:客户端代码

客户端创建接收者、具体命令,并将其注入 Invoker。解耦了“谁发出请求”和“谁执行请求”。

int main() {
    Light livingRoomLight;
LightOnCommand onCmd(livingRoomLight);
LightOffCommand offCmd(livingRoomLight);

RemoteControl remote;

remote.setCommand(&onCmd);
remote.pressButton(); // 输出: Light is ON
remote.pressUndo();   // 输出: Light is OFF

remote.setCommand(&offCmd);
remote.pressButton(); // 输出: Light is OFF
remote.pressUndo();   // 输出: Light is ON

return 0;

}

进阶提示:支持宏命令与智能指针

若需组合多个命令(如“全屋开机”),可定义 MacroCommand,持有一组 std::unique_ptr 并批量执行;使用智能指针避免手动内存管理,尤其在命令需动态创建或生命周期不确定时更安全。

关键点:

  • 命令对象保存执行所需全部上下文(如接收者引用、参数)
  • Invoker 和 Receiver 完全解耦,新增命令无需修改 Invoker
  • 撤销依赖命令自身保存足够状态(如开关前的状态),不是所有操作都天然可逆
  • 实际项目中常配合 std::function 或 lambda 简化一次性命令,但继承体系更适合复杂、需复用或带状态的场景


# mac  # ai  # c++  # 封装  # 子类  # Lambda  # 指针  # 继承  # 接口  # 委托  # function  # 对象  # 调用者  # 客户端  # 进阶  # 多个  # 做什么  # 可以用  # 所需  # 也可  # 可选 


相关栏目: 【 网站优化151355 】 【 网络推广146373 】 【 网络技术251813 】 【 AI营销90571


相关推荐: Python进程池调度策略_任务分发说明【指导】  Laravel模型事件有哪些_Laravel Model Event生命周期详解  *服务器网站为何频现安全漏洞?  深圳网站制作的公司有哪些,dido官方网站?  微信小程序 require机制详解及实例代码  米侠浏览器网页图片不显示怎么办 米侠图片加载修复  ChatGPT 4.0官网入口地址 ChatGPT在线体验官网  如何在万网主机上快速搭建网站?  微信小程序 input输入框控件详解及实例(多种示例)  Windows10电脑怎么查看硬盘通电时间_Win10使用工具检测磁盘健康  Laravel如何实现数据导出到PDF_Laravel使用snappy生成网页快照PDF【方案】  电商网站制作价格怎么算,网上拍卖流程以及规则?  如何用PHP快速搭建CMS系统?  安克发布新款氮化镓充电宝:体积缩小 30%,支持 200W 输出  如何快速查询网址的建站时间与历史轨迹?  Laravel如何使用Eloquent ORM进行数据库操作?(CRUD示例)  Laravel怎么在Controller之外的地方验证数据  uc浏览器二维码扫描入口_uc浏览器扫码功能使用地址  手机网站制作与建设方案,手机网站如何建设?  小视频制作网站有哪些,有什么看国内小视频的网站,求推荐?  laravel怎么用DB facade执行原生SQL查询_laravel DB facade原生SQL执行方法  魔方云NAT建站如何实现端口转发?  免费视频制作网站,更新又快又好的免费电影网站?  专业企业网站设计制作公司,如何理解商贸企业的统一配送和分销网络建设?  佛山网站制作系统,佛山企业变更地址网上办理步骤?  微信h5制作网站有哪些,免费微信H5页面制作工具?  清除minerd进程的简单方法  使用C语言编写圣诞表白程序  中山网站推广排名,中山信息港登录入口?  IOS倒计时设置UIButton标题title的抖动问题  JS去除重复并统计数量的实现方法  网站广告牌制作方法,街上的广告牌,横幅,用PS还是其他软件做的?  Laravel怎么配置.env环境变量_Laravel生产环境敏感数据保护与读取【方法】  百度浏览器ai对话怎么关 百度浏览器ai聊天窗口隐藏  高端建站三要素:定制模板、企业官网与响应式设计优化  如何在景安云服务器上绑定域名并配置虚拟主机?  Linux虚拟化技术教程_KVMQEMU虚拟机安装与调优  Windows10怎样连接蓝牙设备_Windows10蓝牙连接步骤【教程】  如何在IIS中配置站点IP、端口及主机头?  电视网站制作tvbox接口,云海电视怎样自定义添加电视源?  如何快速配置高效服务器建站软件?  什么是javascript作用域_全局和局部作用域有什么区别?  如何自定义safari浏览器工具栏?个性化设置safari浏览器界面教程【技巧】  北京网站制作费用多少,建立一个公司网站的费用.有哪些部分,分别要多少钱?  Laravel策略(Policy)如何控制权限_Laravel Gates与Policies实现用户授权  如何在不使用负向后查找的情况下匹配特定条件前的换行符  linux top下的 minerd 木马清除方法  Laravel如何使用Collections进行数据处理?(实用方法示例)  免费网站制作appp,免费制作app哪个平台好?  Google浏览器为什么这么卡 Google浏览器提速优化设置步骤【方法】