备忘录模式在 golang 中可通过闭包实现,用于保存和恢复对象状态而不破坏封装性。1. 通过结构体方法返回闭包函数捕获并保存状态;2. 闭包调用时还原状态,避免显式定义 memento 结构;3. 常用于撤销操作、游戏存档、事务处理等场景;4. 注意性能问题、避免保存指针、可用 slice 管理多个快照;5. 对于复杂状态或长期服务需谨慎使用,必要时应采用深度拷贝或显式结构体。这种方式简洁灵活,但不适用于所有情况。

在 Golang 中实现备忘录模式,利用闭包保存对象状态是一个既实用又巧妙的做法。备忘录模式的核心在于保存和恢复对象的状态而不破坏其封装性。Golang 虽然没有类的概念,但通过结构体、函数和闭包的组合,可以很好地模拟这一设计模式。

什么是备忘录模式?
简单来说,备忘录模式允许你“记住”一个对象的某个状态,并在之后将其恢复。这种模式常用于撤销操作、版本控制或者快照机制中。

传统的做法是创建一个专门的
Memento类来保存状态,然后由一个管理者来维护这些快照。但在 Golang 中,我们可以借助闭包来更简洁地完成这个任务。
立即学习“go语言免费学习笔记(深入)”;
如何用闭包保存对象状态
闭包在 Go 中非常强大,它可以捕获并保存变量的状态。我们可以通过返回一个函数来“记住”某个结构体的状态,从而避免显式定义 Memento 结构。

举个例子:
type TextEditor struct {
content string
}
func (t *TextEditor) Save() func() {
snapshot := t.content
return func() {
t.content = snapshot
}
}在这个例子中,
Save()方法返回一个闭包函数,该函数会在调用时将当前内容还原为保存时的状态。这种方式省去了单独定义 Memento 结构的麻烦。
使用场景与注意事项
常见使用场景:
- 实现撤销/重做功能(如文本编辑器)
- 游戏中的存档与读档机制
- 状态回滚或事务处理
注意事项:
- 如果对象状态很大或频繁保存,要考虑性能问题
- 不要保存指针类型的状态,否则闭包会引用原始数据,导致状态未真正“冻结”
- 可以配合 map 或 slice 来管理多个快照
例如,你可以这样管理多个快照:
snapshots := make([]func(), 0)
editor := &TextEditor{content: "初始内容"}
snapshots = append(snapshots, editor.Save())
editor.content = "修改后的内容"
snapshots = append(snapshots, editor.Save())
editor.content = "再次修改"
// 恢复到第二个快照
snapshots[1]()
fmt.Println(editor.content) // 输出:修改后的内容是否适合所有情况?
虽然用闭包实现备忘录模式很简洁,但也有一些限制:
- 如果对象状态复杂(嵌套结构体、包含方法等),可能需要深度拷贝
- 对于长期运行的服务,要注意内存管理,及时清理不再需要的快照
如果你只是想快速保存和恢复某个字段的状态,闭包方式非常合适;但如果涉及大量状态或跨结构体的数据,还是建议定义清晰的 Memento 结构。
基本上就这些。闭包让 Go 的备忘录模式实现变得轻量又灵活,理解清楚原理后,在实际项目中可以很方便地应用。










