0

0

C++如何实现命令模式(Command Pattern)_C++设计模式与命令模式实现

冰火之心

冰火之心

发布时间:2025-11-08 18:01:02

|

448人浏览过

|

来源于php中文网

原创

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

c++如何实现命令模式(command pattern)_c++设计模式与命令模式实现

命令模式是一种行为设计模式,它将请求封装为对象,从而使你可以用不同的请求、队列、日志来参数化其他对象。在C++中,命令模式通过抽象命令接口、具体命令类和接收者(Receiver)协作实现解耦。这种模式常用于实现撤销/重做、任务队列、远程调用等场景。

命令模式的核心结构

命令模式通常包含以下几个角色:

  • Command(命令接口):声明执行操作的接口,通常是一个虚函数execute()
  • ConcreteCommand(具体命令):实现Command接口,持有一个接收者对象,并在execute()中调用接收者的相应方法。
  • Receiver(接收者):真正执行请求的对象,包含具体的业务逻辑。
  • Invoker(调用者):持有命令对象,通过调用命令的execute()来触发请求。
  • Client(客户端):创建命令对象并绑定接收者,然后将命令交给调用者。

基础实现示例

以下是一个简单的文本编辑器中“打开文件”和“保存文件”命令的实现:

#include 
#include 
#include 

// 接收者:执行实际操作
class TextEditor {
public:
    void open() {
        std::cout << "TextEditor: 打开文件\n";
    }

    void save() {
        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 SaveCommand : public Command {
private:
    TextEditor& editor;
public:
    SaveCommand(TextEditor& e) : editor(e) {}

    void execute() override {
        editor.save();
    }
};

// 调用者:按钮或快捷键
class Button {
private:
    std::unique_ptr command;
public:
    void setCommand(std::unique_ptr cmd) {
        command = std::move(cmd);
    }

    void click() {
        if (command) {
            command->execute();
        }
    }
};

使用方式如下:

立即学习C++免费学习笔记(深入)”;

问小白
问小白

免费使用DeepSeek满血版

下载
int main() {
    TextEditor editor;

    // 客户端创建命令并绑定接收者
    auto openCmd = std::make_unique(editor);
    auto saveCmd = std::make_unique(editor);

    // 设置到调用者(如按钮)
    Button openButton, saveButton;
    openButton.setCommand(std::move(openCmd));
    saveButton.setCommand(std::move(saveCmd));

    // 模拟用户点击
    openButton.click();  // 输出:TextEditor: 打开文件
    saveButton.click();  // 输出:TextEditor: 保存文件

    return 0;
}

支持撤销操作的扩展

命令模式天然适合实现撤销功能。只需在命令接口中增加undo()方法:

class Command {
public:
    virtual ~Command() = default;
    virtual void execute() = 0;
    virtual void undo() {}  // 可选,默认为空
};

class SaveCommand : public Command {
private:
    TextEditor& editor;
public:
    SaveCommand(TextEditor& e) : editor(e) {}

    void execute() override {
        editor.save();
    }

    void undo() override {
        std::cout << "TextEditor: 撤销保存\n";
    }
};

调用者可以记录历史命令以支持撤销:

class CommandHistory {
private:
    std::vector> history;
public:
    void push(std::unique_ptr cmd) {
        history.push_back(std::move(cmd));
    }

    void undo() {
        if (!history.empty()) {
            history.back()->undo();
            history.pop_back();
        }
    }
};

使用函数对象简化实现

C++11以后,可以用std::function和lambda表达式简化命令模式:

#include 

class FunctionalCommand : public Command {
private:
    std::function doAction;
    std::function undoAction;
public:
    FunctionalCommand(std::function exec, std::function un = []{})
        : doAction(exec), undoAction(un) {}

    void execute() override {
        doAction();
    }

    void undo() override {
        undoAction();
    }
};

// 使用示例
TextEditor editor;
auto cmd = std::make_unique(
    [&editor]() { editor.save(); },
    []() { std::cout << "撤销保存\n"; }
);

基本上就这些。命令模式通过将操作封装成对象,实现了调用者与接收者的解耦,增强了系统的可扩展性和灵活性。不复杂但容易忽略的是,每个命令应尽量保持独立和无状态,避免共享数据带来的副作用。

相关专题

更多
lambda表达式
lambda表达式

Lambda表达式是一种匿名函数的简洁表示方式,它可以在需要函数作为参数的地方使用,并提供了一种更简洁、更灵活的编码方式,其语法为“lambda 参数列表: 表达式”,参数列表是函数的参数,可以包含一个或多个参数,用逗号分隔,表达式是函数的执行体,用于定义函数的具体操作。本专题为大家提供lambda表达式相关的文章、下载、课程内容,供大家免费下载体验。

204

2023.09.15

python lambda函数
python lambda函数

本专题整合了python lambda函数用法详解,阅读专题下面的文章了解更多详细内容。

190

2025.11.08

Python lambda详解
Python lambda详解

本专题整合了Python lambda函数相关教程,阅读下面的文章了解更多详细内容。

49

2026.01.05

硬盘接口类型介绍
硬盘接口类型介绍

硬盘接口类型有IDE、SATA、SCSI、Fibre Channel、USB、eSATA、mSATA、PCIe等等。详细介绍:1、IDE接口是一种并行接口,主要用于连接硬盘和光驱等设备,它主要有两种类型:ATA和ATAPI,IDE接口已经逐渐被SATA接口;2、SATA接口是一种串行接口,相较于IDE接口,它具有更高的传输速度、更低的功耗和更小的体积;3、SCSI接口等等。

1027

2023.10.19

PHP接口编写教程
PHP接口编写教程

本专题整合了PHP接口编写教程,阅读专题下面的文章了解更多详细内容。

66

2025.10.17

php8.4实现接口限流的教程
php8.4实现接口限流的教程

PHP8.4本身不内置限流功能,需借助Redis(令牌桶)或Swoole(漏桶)实现;文件锁因I/O瓶颈、无跨机共享、秒级精度等缺陷不适用高并发场景。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

455

2025.12.29

java接口相关教程
java接口相关教程

本专题整合了java接口相关内容,阅读专题下面的文章了解更多详细内容。

11

2026.01.19

function是什么
function是什么

function是函数的意思,是一段具有特定功能的可重复使用的代码块,是程序的基本组成单元之一,可以接受输入参数,执行特定的操作,并返回结果。本专题为大家提供function是什么的相关的文章、下载、课程内容,供大家免费下载体验。

479

2023.08.04

excel表格操作技巧大全 表格制作excel教程
excel表格操作技巧大全 表格制作excel教程

Excel表格操作的核心技巧在于 熟练使用快捷键、数据处理函数及视图工具,如Ctrl+C/V(复制粘贴)、Alt+=(自动求和)、条件格式、数据验证及数据透视表。掌握这些可大幅提升数据分析与办公效率,实现快速录入、查找、筛选和汇总。

0

2026.01.21

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
10分钟--Midjourney创作自己的漫画
10分钟--Midjourney创作自己的漫画

共1课时 | 0.1万人学习

Midjourney 关键词系列整合
Midjourney 关键词系列整合

共13课时 | 0.9万人学习

AI绘画教程
AI绘画教程

共2课时 | 0.2万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号