备忘录模式在 go 语言中用于保存并恢复对象状态,结合 gob 序列化可高效实现该功能。1. 使用 gob.newencoder 将对象序列化为字节流以保存状态;2. 利用 gob.newdecoder 对字节流进行反序列化以恢复状态;3. 示例中 editor 结构通过 save 和 restore 方法完成内容回滚;4. 根据需求选择内存或文件存储备忘录数据;5. 注意 gob 对结构体字段敏感,重构可能导致旧数据无法解析,需考虑版本兼容性问题。整个实现依赖结构体与接口模拟备忘录模式的三个核心角色:originator、memento 和 caretaker。

在 Go 语言中实现备忘录模式(Memento Pattern),主要是为了保存对象的某个状态,以便后续可以恢复。结合 gob 序列化方案来实现数据恢复,是一个既实用又高效的思路。因为 gob 是 Go 自带的序列化库,适合用于本地状态的持久化和恢复。

下面我们就来看看如何用 Golang + gob 实现一个简单的备忘录模式,并完成数据恢复功能。

什么是备忘录模式?
备忘录模式的核心思想是:在不破坏对象封装性的前提下,捕获并外部化其内部状态,这样之后可以将对象恢复到之前的状态。这个模式常用于撤销操作、版本控制等场景。
立即学习“go语言免费学习笔记(深入)”;
通常来说,它包括三个角色:

- Originator:原始对象,能创建备忘录也能从备忘录恢复。
- Memento:备忘录对象,用来保存 Originator 的状态。
- Caretaker:负责管理备忘录,但不能修改其内容。
在 Go 中,我们可以借助结构体和接口来模拟这些角色。
如何用 gob 序列化保存状态?
Go 的标准库 encoding/gob 可以将结构体进行二进制编码,非常适合用于保存对象状态。相比 JSON,它的性能更好,而且对结构体嵌套支持更自然。
我们可以通过如下方式保存对象状态:
func saveState(obj any, filename string) error {
file, err := os.Create(filename)
if err != nil {
return err
}
defer file.Close()
encoder := gob.NewEncoder(file)
return encoder.Encode(obj)
}这段代码会把传入的对象序列化后写入文件,作为“快照”保存下来。
如何从 gob 快照恢复数据?
恢复过程就是反序列化的过程,只需要读取文件并解码即可:
func restoreState(obj any, filename string) error {
file, err := os.Open(filename)
if err != nil {
return err
}
defer file.Close()
decoder := gob.NewDecoder(file)
return decoder.Decode(obj)
}注意,这里的 obj 要传入指针类型,这样才能正确赋值。
示例:用备忘录模式实现简单回滚
假设我们有一个编辑器类结构,记录了当前的内容:
type Editor struct {
Content string
}
// 创建备忘录
func (e *Editor) Save() ([]byte, error) {
var buf bytes.Buffer
encoder := gob.NewEncoder(&buf)
err := encoder.Encode(e)
return buf.Bytes(), err
}
// 从备忘录恢复
func (e *Editor) Restore(data []byte) error {
buf := bytes.NewBuffer(data)
decoder := gob.NewDecoder(buf)
return decoder.Decode(e)
}然后,我们可以在用户操作前后保存状态:
editor := &Editor{Content: "初始文本"}
memento, _ := editor.Save()
editor.Content = "修改后的文本"
fmt.Println("当前内容:", editor.Content)
editor.Restore(memento)
fmt.Println("恢复后内容:", editor.Content)输出结果就会是:
当前内容: 修改后的文本 恢复后内容: 初始文本
小贴士:使用文件还是内存?按需选择
- 如果你只是临时保存几个状态,比如做撤销/重做功能,可以直接用字节数组存到内存里。
- 如果需要长期保存或者跨程序运行恢复,就用文件存储更合适。
- 注意 gob 编码的数据是二进制格式,不适合直接查看或传输给其他语言系统。
另外,gob 在编解码时对结构体字段名敏感,如果你重构了结构体,旧数据可能会无法解析。所以在实际项目中,如果要做版本兼容,可能还需要配合别的机制。
基本上就这些。用 Golang 实现备忘录模式的关键在于状态的保存与还原逻辑,而 gob 提供了一个轻量又高效的手段。虽然看起来不复杂,但在实际开发中很容易忽略一些细节,比如指针传递、结构一致性等问题,建议在使用前多做测试。











