0

0

如何用Golang实现备忘录模式 保存和恢复对象状态技巧

P粉602998670

P粉602998670

发布时间:2025-08-17 12:18:02

|

586人浏览过

|

来源于php中文网

原创

答案:备忘录模式通过私有状态字段、深拷贝和序列化保障封装性,适用于撤销/重做、游戏存档等场景。

如何用golang实现备忘录模式 保存和恢复对象状态技巧

在Golang中实现备忘录模式,核心在于定义一个“备忘录”结构体来封装对象的状态,一个“发起人”对象负责创建和恢复这些备忘录,以及一个“看管者”来管理这些备忘录的历史记录。这种模式对于实现撤销/重做功能、保存游戏进度或任何需要回溯对象状态的场景都极其有效。

解决方案

备忘录模式(Memento Pattern)旨在不破坏封装性的前提下,捕获一个对象的内部状态,并在该对象之外保存这个状态。这样,以后就可以将该对象恢复到原先保存的状态。

在Golang中,我们可以这样构建它:

  1. 备忘录(Memento):一个结构体,用于存储发起人对象的内部状态。它通常只包含数据,没有复杂的行为。

    立即学习go语言免费学习笔记(深入)”;

    type Memento struct {
        State string // 假设状态是一个字符串,实际可以是更复杂的结构
        // 也可以包含其他需要保存的状态字段
    }
  2. 发起人(Originator):需要保存其状态的对象。它负责创建备忘录(保存当前状态)和使用备忘录恢复状态。

    type Originator struct {
        current string // 发起人的当前状态
    }
    
    func (o *Originator) SetState(state string) {
        o.current = state
        // fmt.Printf("Originator: Setting state to %s\n", state)
    }
    
    func (o *Originator) SaveStateToMemento() *Memento {
        // fmt.Printf("Originator: Saving state %s to Memento.\n", o.current)
        return &Memento{State: o.current}
    }
    
    func (o *Originator) RestoreStateFromMemento(m *Memento) {
        o.current = m.State
        // fmt.Printf("Originator: State restored to %s from Memento.\n", o.current)
    }
    
    func (o *Originator) GetState() string {
        return o.current
    }
  3. 看管者(Caretaker):负责保存和管理备忘录。它从不检查备忘录的内容。

    type Caretaker struct {
        mementoList []*Memento
    }
    
    func (c *Caretaker) AddMemento(m *Memento) {
        c.mementoList = append(c.mementoList, m)
    }
    
    func (c *Caretaker) GetMemento(index int) *Memento {
        if index >= 0 && index < len(c.mementoList) {
            return c.mementoList[index]
        }
        return nil // 或者返回错误
    }

示例用法:

// main.go
package main

import "fmt"

func main() {
    originator := &Originator{}
    caretaker := &Caretaker{}

    originator.SetState("State #1")
    caretaker.AddMemento(originator.SaveStateToMemento())

    originator.SetState("State #2")
    caretaker.AddMemento(originator.SaveStateToMemento())

    originator.SetState("State #3")
    fmt.Printf("Current State: %s\n", originator.GetState()) // 应该是 State #3

    originator.RestoreStateFromMemento(caretaker.GetMemento(0))
    fmt.Printf("First saved State: %s\n", originator.GetState()) // 应该是 State #1

    originator.RestoreStateFromMemento(caretaker.GetMemento(1))
    fmt.Printf("Second saved State: %s\n", originator.GetState()) // 应该是 State #2
}

备忘录模式在Golang中如何保障对象状态的封装性?

备忘录模式的精髓在于,它允许在不暴露对象内部细节的前提下,外部化并保存对象的状态。在Golang里,这主要通过以下几个方面实现:

首先,

Originator
(发起人)内部的状态字段通常是私有的(小写字母开头),外部无法直接访问。当
Originator
创建
Memento
时,它会将其内部状态的副本传递给
Memento
。而
Caretaker
(看管者)只持有
Memento
的引用,它并不知道
Memento
内部具体存储了什么,也无法直接修改
Originator
的私有状态。它只能通过
Originator
提供的公共方法(如
RestoreStateFromMemento
)来间接操作。

我个人在实践中发现,这种模式的封装性在处理复杂类型时尤其需要注意。如果

Originator
的状态包含切片、映射或指针等引用类型,仅仅将它们赋值给
Memento
字段,实际上只是复制了引用。这意味着如果
Originator
后续修改了这些引用类型指向的数据,
Memento
中的“保存”状态也会跟着改变,这显然违背了备忘录模式的初衷。为了真正保障封装和状态的独立性,
SaveStateToMemento
方法内部必须执行“深拷贝”(Deep Copy),确保
Memento
持有的是状态数据的完全独立副本,而不是共享引用。这虽然增加了实现的复杂性,但对于确保状态的完整性和隔离性至关重要。

HTTPie AI
HTTPie AI

AI API开发工具

下载

Golang实现备忘录模式时,如何处理复杂或大型对象的状态?

处理复杂或大型对象的状态是备忘录模式在实际应用中常常遇到的挑战。简单地复制整个对象可能导致内存消耗过大,尤其是在需要保存大量历史状态时。

一种常见的策略是序列化。如果对象状态非常复杂,或者需要跨进程、跨网络传输,甚至持久化到磁盘,那么将状态序列化成字节流(例如JSON、Gob、Protocol Buffers)存储在

Memento
中是一个非常实用的方法。
Memento
可以只包含一个
[]byte
字段,
Originator
在保存时将自身状态序列化,在恢复时再反序列化。这使得
Memento
的结构变得非常简单,但代价是序列化和反序列化的性能开销。我曾经在一个项目中,由于需要保存一个包含大量嵌套结构和自定义类型的数据模型,直接深拷贝的成本太高,最终选择了
Gob
序列化,效果非常不错。

另一种思路是部分状态保存。如果一个对象的完整状态非常庞大,但只有其中一小部分是经常变化的,并且足以恢复整个对象,那么

Memento
可以只保存这部分关键状态。例如,一个大型UI组件可能有很多瞬态属性,但只有其配置和数据源是需要保存的。

对于频繁变动但每次变动都很小的场景,可以考虑增量备忘录(Delta Memento)。不是保存完整的状态,而是只保存与前一个状态之间的“差异”或“操作日志”。恢复时,需要从初始状态开始,逐个应用这些差异。这种方法在内存效率上非常有优势,但在实现上会复杂得多,因为需要一个机制来计算差异和应用差异。这通常与命令模式结合使用,将每个操作本身作为一个可撤销的“命令”来存储。

备忘录模式在Golang的哪些实际场景中特别有用?

备忘录模式的应用场景远不止于经典的“撤销/重做”功能,它在很多需要“时间旅行”或者“状态快照”的系统中都扮演着关键角色。

最直观的,当然是图形编辑器或文本编辑器中的撤销/重做功能。用户每执行一个操作(如输入文字、移动图形),系统就保存当前文档的一个快照(备忘录),并将其压入一个历史栈。当用户点击“撤销”时,就从栈中取出前一个备忘录来恢复文档状态。

游戏开发中,备忘录模式是实现保存/加载游戏进度的基石。玩家可以在任何时候保存游戏状态,下次启动时可以从上次保存的点继续。每个保存点就是一个备忘录。

此外,在一些业务系统或数据处理流程中,备忘录模式也很有用。例如,一个复杂的数据转换或计算流程,可能需要分多个步骤进行。如果在某个步骤出现错误,我们可能需要将系统回滚到之前的某个稳定状态,然后重新尝试。备忘录模式可以提供这种事务性回滚的能力,虽然通常数据库事务是更常见的解决方案,但在内存中的复杂对象状态管理上,备忘录模式能提供轻量级的方案。

我也在一些配置管理系统中看到它的身影。当配置项发生变更时,可以自动保存一个旧版本的配置备忘录。这样,如果新配置导致问题,可以快速回滚到之前的稳定配置。这在不中断服务的前提下进行配置变更时,提供了一层重要的安全保障。它提供了一种灵活且相对独立的机制来管理对象状态的历史,避免了将状态管理逻辑与业务逻辑耦合在一起。

相关专题

更多
golang如何定义变量
golang如何定义变量

golang定义变量的方法:1、声明变量并赋予初始值“var age int =值”;2、声明变量但不赋初始值“var age int”;3、使用短变量声明“age :=值”等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

174

2024.02.23

golang有哪些数据转换方法
golang有哪些数据转换方法

golang数据转换方法:1、类型转换操作符;2、类型断言;3、字符串和数字之间的转换;4、JSON序列化和反序列化;5、使用标准库进行数据转换;6、使用第三方库进行数据转换;7、自定义数据转换函数。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

224

2024.02.23

golang常用库有哪些
golang常用库有哪些

golang常用库有:1、标准库;2、字符串处理库;3、网络库;4、加密库;5、压缩库;6、xml和json解析库;7、日期和时间库;8、数据库操作库;9、文件操作库;10、图像处理库。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

335

2024.02.23

golang和python的区别是什么
golang和python的区别是什么

golang和python的区别是:1、golang是一种编译型语言,而python是一种解释型语言;2、golang天生支持并发编程,而python对并发与并行的支持相对较弱等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

206

2024.03.05

golang是免费的吗
golang是免费的吗

golang是免费的。golang是google开发的一种静态强类型、编译型、并发型,并具有垃圾回收功能的开源编程语言,采用bsd开源协议。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

388

2024.05.21

golang结构体相关大全
golang结构体相关大全

本专题整合了golang结构体相关大全,想了解更多内容,请阅读专题下面的文章。

193

2025.06.09

golang相关判断方法
golang相关判断方法

本专题整合了golang相关判断方法,想了解更详细的相关内容,请阅读下面的文章。

188

2025.06.10

golang数组使用方法
golang数组使用方法

本专题整合了golang数组用法,想了解更多的相关内容,请阅读专题下面的文章。

191

2025.06.17

小游戏4399大全
小游戏4399大全

4399小游戏免费秒玩大全来了!无需下载、即点即玩,涵盖动作、冒险、益智、射击、体育、双人等全品类热门小游戏。经典如《黄金矿工》《森林冰火人》《狂扁小朋友》一应俱全,每日更新最新H5游戏,支持电脑与手机跨端畅玩。访问4399小游戏中心,重温童年回忆,畅享轻松娱乐时光!官方入口安全绿色,无插件、无广告干扰,打开即玩,快乐秒达!

30

2025.12.31

热门下载

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

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
golang socket 编程
golang socket 编程

共2课时 | 0.1万人学习

nginx浅谈
nginx浅谈

共15课时 | 0.8万人学习

golang和swoole核心底层分析
golang和swoole核心底层分析

共3课时 | 0.1万人学习

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

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