命令模式实现撤销重做,关键在操作记录与回退
时间:2025-10-26 09:52:49 153浏览 收藏
小伙伴们有没有觉得学习文章很有意思?有意思就对了!今天就给大家带来《命令模式实现撤销重做功能,关键在于记录操作历史并提供回退机制。以下是实现步骤和示例代码:一、核心概念Command 接口 定义执行和撤销方法。ConcreteCommand(具体命令) 实现 Command 接口,保存需要执行的操作和参数。Invoker(调用者) 负责触发命令,并维护命令的历史记录。Receiver(接收者) 执行具体操作的对象。History(历史记录) 存储已执行的命令,用于撤销或重做。二、实现步骤1. 定义 Command 接口class Command { execute() {} undo() {} }2. 创建具体命令类(如:加法、减法) class AddCommand extends Command { constructor(receiver, value) { super(); this.receiver = receiver; this.value = value; } execute() { this.receiver.add(this.value); } undo() { this.receiver.subtract(this.value); } } class SubtractCommand extends Command { constructor(receiver, value) { super(); this.receiver = receiver; this.value = value; } execute() { this.receiver.subtract(this.value);》,以下内容将会涉及到,若是在学习中对其中部分知识点有疑问,或许看了本文就能帮到你!
实现撤销和重做功能的关键是将操作封装为命令对象,通过历史栈管理执行与反向逻辑。1. 定义命令类如SetTextCommand,保存执行前后的状态;2. 创建CommandManager管理undo和redo栈,执行命令时压入undo栈并清空redo栈;3. 调用undo时将命令从undo栈弹出,执行反操作后压入redo栈;4. redo则反向操作,恢复已撤销的命令。示例中文本编辑器通过该模式实现内容修改、撤销与重做。扩展可支持复合命令、限制栈大小、添加命令描述等。核心在于动作对象化与栈的顺序控制,注意清空redo栈的时机以保证操作一致性。

实现一个支持撤销和重做的命令模式,关键在于把每个操作封装成对象,记录执行和反向操作的逻辑,并通过历史栈管理这些命令。下面是一个简洁、实用的实现方式。
定义命令接口
每个命令需要统一结构,包含执行(execute)、撤销(undo)方法。这样能保证调用一致性。
例如,假设我们在做一个简单的文本编辑器,可以修改内容:
class SetTextCommand {
constructor(editor, newText) {
this.editor = editor;
this.newText = newText;
this.oldText = editor.text; // 保存之前的状态用于撤销
}
execute() {
this.editor.text = this.newText;
}
undo() {
this.editor.text = this.oldText;
}
}
创建命令管理器
命令管理器负责执行命令,并维护撤销和重做栈。
class CommandManager {
constructor() {
this.undoStack = [];
this.redoStack = [];
}
execute(command) {
command.execute();
this.undoStack.push(command);
this.redoStack = []; // 执行新命令后,清空重做栈
}
undo() {
if (this.undoStack.length === 0) return;
const command = this.undoStack.pop();
command.undo();
this.redoStack.push(command);
}
redo() {
if (this.redoStack.length === 0) return;
const command = this.redoStack.pop();
command.execute();
this.undoStack.push(command);
}
}
使用示例
将命令模式应用到具体场景中:
const editor = { text: "" };
const commandManager = new CommandManager();
// 修改文本
commandManager.execute(new SetTextCommand(editor, "Hello"));
console.log(editor.text); // 输出: Hello
commandManager.execute(new SetTextCommand(editor, "Hello World"));
console.log(editor.text); // 输出: Hello World
// 撤销
commandManager.undo();
console.log(editor.text); // 输出: Hello
// 重做
commandManager.redo();
console.log(editor.text); // 输出: Hello World
扩展建议
实际项目中可进一步优化:
- 支持批量命令(CompositeCommand),把多个命令组合成一个可撤销的操作
- 限制历史栈大小,避免内存泄漏
- 为命令添加描述,便于实现 UI 操作历史列表
- 异步命令需额外处理,比如保存状态快照或使用回调机制
基本上就这些。核心是把“动作”变成对象,控制执行流程,再用栈管理顺序。不复杂但容易忽略细节,比如清空重做栈的时机。
以上就是本文的全部内容了,是否有顺利帮助你解决问题?若是能给你带来学习上的帮助,请大家多多支持golang学习网!更多关于文章的相关知识,也可关注golang学习网公众号。
-
502 收藏
-
501 收藏
-
501 收藏
-
501 收藏
-
501 收藏
-
179 收藏
-
122 收藏
-
404 收藏
-
201 收藏
-
182 收藏
-
407 收藏
-
324 收藏
-
330 收藏
-
465 收藏
-
204 收藏
-
390 收藏
-
464 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 立即学习 543次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 立即学习 516次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 立即学习 500次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 立即学习 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 立即学习 485次学习