0

0

如何在Golang中实现备忘录模式历史记录_Golang备忘录模式状态保存方法

P粉602998670

P粉602998670

发布时间:2025-12-30 18:16:02

|

453人浏览过

|

来源于php中文网

原创

Go 语言可通过结构体值拷贝、闭包或 JSON 序列化模拟备忘录模式,核心是安全保存与恢复对象状态而不破坏封装;需深拷贝避免引用污染,备忘录应不可变且字段小写,历史栈需管理索引与容量。

如何在golang中实现备忘录模式历史记录_golang备忘录模式状态保存方法

Go 语言没有类和继承,也不支持传统面向对象语境下的「备忘录模式」UML 实现(比如 Memento 接口 + Originator + Caretaker 三角色),但完全可以用结构体、值语义和闭包模拟出等效行为——关键是理解它要解决的问题:**安全地保存和恢复某个对象的内部状态快照,且不破坏封装**。

用结构体字段 + 值拷贝实现最简备忘录

Go 中最自然的方式是让原对象(Originator)自己提供 Save()Restore() 方法,返回/接收一个只含必要字段的纯数据结构(即备忘录)。这个结构体不暴露内部逻辑,仅作序列化载体。

常见错误是直接保存指针或 map/slice 引用,导致快照被后续修改污染。

  • Save() 必须做深拷贝:对 mapslice、嵌套结构体手动复制,或用 encoding/gob / json 编解码(注意非导出字段会被忽略)
  • 备忘录结构体字段应全小写(如 stateversion),避免外部直接访问,靠 Originator 控制读写权限
  • 不要把备忘录设计成可变结构;它应是不可变快照,每次 Save() 返回新实例
type Editor struct {
	content string
	cursor  int
}

type editorMemento struct {
	content string
	cursor  int
}

func (e *Editor) Save() editorMemento {
	return editorMemento{
		content: e.content,
		cursor:  e.cursor,
	}
}

func (e *Editor) Restore(m editorMemento) {
	e.content = m.content
	e.cursor = m.cursor
}

用切片管理历史:Undo/Redo 场景

当需要多步回退(Undo)和重做(Redo),备忘录需存入栈式结构。关键点是控制容量、避免内存泄漏、区分「当前态」与「快照态」。

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

典型陷阱是未清空 redo 栈:每次新操作后,用户若执行了 Undo 再输入新内容,旧的 Redo 链必须截断。

Fireflies.ai
Fireflies.ai

自动化会议记录和笔记工具,可以帮助你的团队记录、转录、搜索和分析语音对话。

下载
  • 用两个切片:history []editorMemento 存所有已提交快照,index int 指向当前有效位置(类似游标)
  • 每次 Save() 后,截断 history[index+1:],再追加新快照,并更新 index
  • Undo()index 减 1 并 Restore()Redo() 则加 1(需检查边界)
  • 为防爆内存,可限制 history 最大长度,超出时从头部裁剪(history = history[1:]

用闭包封装状态 + 备忘录工厂函数

若不想暴露结构体定义,可用闭包将状态和操作函数打包,对外只返回操作接口。这种方式更贴近「封装内部状态」的原始意图。

缺点是无法跨 goroutine 共享备忘录,且调试时难追踪字段变化。

  • 工厂函数返回 Save()Restore() 闭包,共享同一份状态变量
  • 备忘录本身是匿名结构字面量或 map[string]interface{},但务必确保值拷贝
  • 慎用 unsafe.Pointer 或反射做「伪深拷贝」——极易出错,且失去类型安全
func NewEditor() (save func() map[string]interface{}, restore func(map[string]interface{})) {
	state := map[string]interface{}{
		"content": "",
		"cursor":  0,
	}

	save = func() map[string]interface{} {
		cp := make(map[string]interface{})
		for k, v := range state {
			cp[k] = v // 注意:此处仅浅拷贝;若 v 是 map/slice,需递归处理
		}
		return cp
	}

	restore = func(m map[string]interface{}) {
		state["content"] = m["content"]
		state["cursor"] = m["cursor"]
	}

	return save, restore
}

JSON 序列化备忘录的兼容性坑

json.Marshal / json.Unmarshal 管理备忘录看似方便,但实际有多个隐性约束:

  • 结构体字段必须首字母大写(否则 JSON 包忽略),但这样又破坏了「内部状态封装」的设计初衷
  • 时间类型、自定义类型、函数字段无法直接序列化,需实现 MarshalJSON 方法
  • 浮点数精度丢失、NaN / Inf 会触发错误,需预检
  • 性能开销比纯结构体拷贝高约 3–5 倍,高频操作(如编辑器实时快照)慎用

真正需要跨进程/持久化时再上 JSON;单进程内状态快照,优先用结构体值拷贝。

相关专题

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

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

173

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相关判断方法,想了解更详细的相关内容,请阅读下面的文章。

187

2025.06.10

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

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

191

2025.06.17

桌面文件位置介绍
桌面文件位置介绍

本专题整合了桌面文件相关教程,阅读专题下面的文章了解更多内容。

0

2025.12.30

热门下载

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

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
WEB前端教程【HTML5+CSS3+JS】
WEB前端教程【HTML5+CSS3+JS】

共101课时 | 8.1万人学习

JS进阶与BootStrap学习
JS进阶与BootStrap学习

共39课时 | 3.1万人学习

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

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