0

0

如何实现一个支持撤销重做的命令模式历史管理器?

夢幻星辰

夢幻星辰

发布时间:2025-10-08 09:37:01

|

787人浏览过

|

来源于php中文网

原创

答案:通过命令模式将操作封装为对象,利用历史栈和重做栈实现撤销与重做功能。具体操作实现execute和undo方法,HistoryManager管理命令执行、撤销与重做流程,支持文本编辑等可逆操作,并注意合并输入、标记不可撤销命令及避免内存泄漏等问题。

如何实现一个支持撤销重做的命令模式历史管理器?

实现一个支持撤销重做的命令模式历史管理器,核心是将用户的操作封装为“命令”对象,并通过历史记录这些命令的执行顺序。这样可以在不暴露具体业务逻辑的前提下,统一管理操作的撤销(undo)与重做(redo)。

定义命令接口

每个可撤销、可重做的操作都应实现统一的命令接口。该接口至少包含两个方法:

  • execute():执行操作
  • undo():撤销操作

如果需要支持重做,通常在执行新命令时清空重做栈,而重做操作则是将已撤销的命令重新执行。

示例:
class Command {
  execute() {}
  undo() {}
}

维护历史栈

使用两个数组分别存储已执行的命令和已被撤销的命令:

  • history:存放已成功执行的命令(用于撤销)
  • redoStack:存放被撤销的命令(用于重做)

当执行新命令时,将其加入 history 栈,并清空 redoStack;撤销时从 history 弹出命令并推入 redoStack;重做则相反。

萝卜简历
萝卜简历

免费在线AI简历制作工具,帮助求职者轻松完成简历制作。

下载
关键逻辑:
class HistoryManager {
  constructor() {
    this.history = [];
    this.redoStack = [];
  }

  execute(command) {
    command.execute();
    this.history.push(command);
    this.redoStack = []; // 新操作后,重做栈失效
  }

  undo() {
    if (this.history.length === 0) return;
    const command = this.history.pop();
    command.undo();
    this.redoStack.push(command);
  }

  redo() {
    if (this.redoStack.length === 0) return;
    const command = this.redoStack.pop();
    command.execute();
    this.history.push(command);
  }
}

实现具体命令

每一个用户操作(如修改文本、移动元素)都应封装为具体的命令类。命令需保存足够的上下文信息,以便正确执行和撤销。

例子:文本编辑命令
class TextEditCommand extends Command {
  constructor(editor, oldText, newText) {
    super();
    this.editor = editor;
    this.oldText = oldText;
    this.newText = newText;
  }

  execute() {
    this.editor.setText(this.newText);
  }

  undo() {
    this.editor.setText(this.oldText);
  }
}

使用时只需将命令交给 HistoryManager 执行:

const manager = new HistoryManager();
manager.execute(new TextEditCommand(editor, "hello", "world"));
manager.undo(); // 恢复为 "hello"
manager.redo(); // 变回 "world"

处理边界情况

实际应用中需注意几个细节:

  • 连续输入等高频操作可合并为一个命令,避免历史过深
  • 某些命令不可撤销(如删除敏感数据),应在设计时明确标记
  • 命令对象持有对业务对象的引用,注意避免内存泄漏
  • 异步操作需等待完成后再压入历史栈

基本上就这些。只要把操作抽象成命令,再用栈管理执行轨迹,撤销重做就很清晰了。

相关专题

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

硬盘接口类型有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瓶颈、无跨机共享、秒级精度等缺陷不适用高并发场景。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

454

2025.12.29

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

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

10

2026.01.19

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

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

392

2023.07.18

堆和栈区别
堆和栈区别

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

572

2023.08.10

Java JVM 原理与性能调优实战
Java JVM 原理与性能调优实战

本专题系统讲解 Java 虚拟机(JVM)的核心工作原理与性能调优方法,包括 JVM 内存结构、对象创建与回收流程、垃圾回收器(Serial、CMS、G1、ZGC)对比分析、常见内存泄漏与性能瓶颈排查,以及 JVM 参数调优与监控工具(jstat、jmap、jvisualvm)的实战使用。通过真实案例,帮助学习者掌握 Java 应用在生产环境中的性能分析与优化能力。

19

2026.01.20

PS使用蒙版相关教程
PS使用蒙版相关教程

本专题整合了ps使用蒙版相关教程,阅读专题下面的文章了解更多详细内容。

61

2026.01.19

java用途介绍
java用途介绍

本专题整合了java用途功能相关介绍,阅读专题下面的文章了解更多详细内容。

87

2026.01.19

热门下载

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

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
国外Web开发全栈课程全集
国外Web开发全栈课程全集

共12课时 | 1.0万人学习

进程与SOCKET
进程与SOCKET

共6课时 | 0.3万人学习

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

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