0

0

C++如何实现一个命令模式_C++设计模式之请求封装与撤销/重做功能

冰火之心

冰火之心

发布时间:2025-12-02 16:43:59

|

1036人浏览过

|

来源于php中文网

原创

命令模式将请求封装为对象,支持参数化操作与撤销重做。通过Command接口、具体命令类、接收者、调用者和客户端协作,实现文本编辑器的插入删除及历史控制,便于扩展与维护。

c++如何实现一个命令模式_c++设计模式之请求封装与撤销/重做功能

在C++中实现命令模式,核心是将“请求”封装成独立对象,使得可以用不同的请求对客户进行参数化,并支持请求的撤销、重做、排队等操作。这种设计特别适用于需要支持操作历史记录的场景,比如文本编辑器、图形界面操作或游戏中的技能系统。

命令模式的基本结构

命令模式包含以下几个关键角色:

  • Command(命令接口):声明执行操作的接口,通常包含一个execute()方法,有时还包括undo()和redo()。
  • ConcreteCommand(具体命令):实现Command接口,持有对Receiver的引用,并在execute()中调用Receiver的相应方法。
  • Receiver(接收者):真正执行请求的对象,比如文档、画布或游戏角色。
  • Invoker(调用者):持有命令对象,通过调用命令的execute()来触发请求,例如按钮或菜单项。
  • Client(客户端):创建ConcreteCommand对象,并设置其接收者。

举个例子:一个简单的文本编辑器支持“插入文本”和“删除文本”操作,并能撤销上一步。

基础实现:支持执行与撤销

先定义命令基类:

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

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

定义接收者——文本编辑器:

Sora
Sora

Sora是OpenAI发布的一种文生视频AI大模型,可以根据文本指令创建现实和富有想象力的场景。

下载
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; }
};

实现具体命令:

class InsertCommand : public Command {
    TextEditor* editor;
    std::string text;
public:
    InsertCommand(TextEditor* e, const std::string& t) : editor(e), text(t) {}
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> commands;
    int current = -1;  // 当前位置,用于支持redo

public: void execute(std::unique_ptr cmd) { cmd->execute(); // 清除当前位置之后的历史(类似浏览器行为) if (current < (int)commands.size() - 1) { commands.erase(commands.begin() + current + 1, commands.end()); } commands.push_back(std::move(cmd)); current++; }

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;
auto insertHello = std::make_unique(&editor, "Hello");
auto insertWorld = std::make_unique(&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;

}

基本上就这些。通过命令模式,你把“动作”变成了可存储、可传递的对象。它让调用者不依赖具体操作,也方便扩展新命令而不改动现有代码。配合历史,轻松实现撤销/重做功能。对于更复杂的场景,可以加入命令合并(如连续输入合并为一次)、命令序列(宏)、持久化等特性。

相关专题

更多
string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

318

2023.08.02

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

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

1023

2023.10.19

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

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

66

2025.10.17

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

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

438

2025.12.29

堆和栈的区别
堆和栈的区别

堆和栈的区别:1、内存分配方式不同;2、大小不同;3、数据访问方式不同;4、数据的生命周期。本专题为大家提供堆和栈的区别的相关的文章、下载、课程内容,供大家免费下载体验。

391

2023.07.18

堆和栈区别
堆和栈区别

堆(Heap)和栈(Stack)是计算机中两种常见的内存分配机制。它们在内存管理的方式、分配方式以及使用场景上有很大的区别。本文将详细介绍堆和栈的特点、区别以及各自的使用场景。php中文网给大家带来了相关的教程以及文章欢迎大家前来学习阅读。

572

2023.08.10

class在c语言中的意思
class在c语言中的意思

在C语言中,"class" 是一个关键字,用于定义一个类。想了解更多class的相关内容,可以阅读本专题下面的文章。

465

2024.01.03

python中class的含义
python中class的含义

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

13

2025.12.06

高德地图升级方法汇总
高德地图升级方法汇总

本专题整合了高德地图升级相关教程,阅读专题下面的文章了解更多详细内容。

72

2026.01.16

热门下载

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

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
C# 教程
C# 教程

共94课时 | 7万人学习

C 教程
C 教程

共75课时 | 4.1万人学习

C++教程
C++教程

共115课时 | 12.7万人学习

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

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