命令模式与备忘录模式结合的核心是命令对象自带快照能力,执行前自动保存影响的状态字段,通过done/undone双栈实现Undo/Redo,利用Go值语义避免深拷贝,确保快照时机正确和Undo幂等。

命令模式和备忘录模式结合,核心是让每个命令对象自带“快照能力”,执行前自动保存状态,支持撤销(Undo)和重做(Redo)。Golang 没有类继承和接口强制实现,但通过组合、函数字段和结构体嵌套,能自然、轻量地实现这一组合。
用结构体封装命令 + 状态快照
定义一个通用命令接口(实际是函数类型或带 Execute/Undo 方法的结构体),再让每个具体命令持有一个可序列化的状态副本。关键不是“保存整个程序状态”,而是只保存该命令影响的那部分数据。
- 用 struct 字段记录操作前的关键值(如编辑前的文本、移动前的坐标)
- 命令执行时修改外部状态,同时把旧值存入自身字段
- Undo 时直接用这个字段恢复,无需外部存储或反射
用栈管理命令历史(支持多级撤销)
维护两个切片:done(已执行命令栈)、undone(已撤销命令栈)。每次 Execute 后压入 done;Undo 时弹出并调用其 Undo 方法,再压入 undone;Redo 则从 undone 弹出并重新 Execute。
- 初始化:done, undone := make([]*Command, 0), make([]*Command, 0)
- Execute 后:done = append(done, cmd)
- Undo 时:cmd := done[len(done)-1]; done = done[:len(done)-1]; cmd.Undo(); undone = append(undone, cmd)
避免深拷贝:用值语义或浅层复制
Golang 中 struct 默认值传递,只要命令中保存的状态字段是基础类型、数组或小结构体(如 type Vec2 struct{ X, Y float64 }),直接赋值即可安全快照。若涉及 map/slice/指针,需手动 copy 或使用 json.Marshal/Unmarshal 做浅层隔离。
立即学习“go语言免费学习笔记(深入)”;
- 推荐方式:状态字段设计为纯值类型,例如 Text string、Position image.Point
- 不推荐:直接存 *Document 或 map[string]interface{} 而不做拷贝
示例:简易文本编辑器的撤销逻辑
比如 “替换文本” 命令:
type ReplaceCmd struct {
Target *string
Old, New string
}
func (c *ReplaceCmd) Execute() {
c.Old = *c.Target
*c.Target = c.New
}
func (c *ReplaceCmd) Undo() {
*c.Target = c.Old
}
调用方只需:
- 创建命令:cmd := &ReplaceCmd{Target: &doc.Content, New: "hello"}
- 执行:cmd.Execute(); history.Push(cmd)
- 撤销:history.Undo() → 自动恢复原内容
基本上就这些。不复杂但容易忽略的是:快照时机必须在 Execute 前完成,且 Undo 必须是幂等的。Golang 的组合与值语义,反而让这种模式比面向对象语言更清晰、更可控。










