如何在Golang中实现备忘录模式历史记录_Golang备忘录模式状态保存方法
发布时间 - 2025-12-30 00:00:00 点击率:次Go 语言可通过结构体值拷贝、闭包或 JSON 序列化模拟备忘录模式,核心是安全保存与恢复对象状态而不破坏封装;需深拷贝避免引用污染,备忘录应不可变且字段小写,历史栈需管理索引与容量。
Go 语言没有类和继承,也不支持传统面向对象语境下的「备忘录模式」UML 实现(比如 Memento 接口 + Originator + Caretaker 三角色),但完全可以用结构体、值语义和闭包模拟出等效行为——关键是理解它要解决的问题:**安全地保存和恢复某个对象的内部状态快照,且不破坏封装**。
用结构体字段 + 值拷贝实现最简备忘录
Go 中最自然的方式是让原对象(Originator)自己提供 Save() 和 Restore() 方法,返回/接收一个只含必要字段的纯数据结构(即备忘录)。这个结构体不暴露内部逻辑,仅作序列化载体。
常见错误是直接保存指针或 map/slice 引用,导致快照被后续修改污染。
-
Save()必须做深拷贝:对map、slice、嵌套结构体手动复制,或用encoding/gob/json编解码(注意非导出字段会被忽略) - 备忘录结构体字段应全小写(如
state、version),避免外部直接访问,靠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),备忘录需存入栈式结构。关键点是控制容量、避免内存泄漏、区分「当前态」与「快照态」。
典型陷阱是未清空 redo 栈:每次新操作后,用户若执行了 Undo 再输入新内容,旧的 Redo 链必须截断。
- 用两个切片:
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;单进程内状态快照,优先用结构体值拷贝。
# js
# json
# go
# golang
# 栈
# red
# String
# 面向对象
# 封装
# 结构体
# int
# 指针
# 数据结构
# 继承
# 接口
# Interface
# 闭包
# pointer
# 切片
# map
# 对象
# history
# uml
# 序列化
# 递归
# 也不
# 多个
# 可以用
# 要把
# 而不
# 自定义
# 可通过
相关栏目:
【
网站优化151355 】
【
网络推广146373 】
【
网络技术251813 】
【
AI营销90571 】
相关推荐:
如何自己制作一个网站链接,如何制作一个企业网站,建设网站的基本步骤有哪些?
Claude怎样写约束型提示词_Claude约束提示词写法【教程】
郑州企业网站制作公司,郑州招聘网站有哪些?
PHP怎么接收前端传的文件路径_处理文件路径参数接收方法【汇总】
Win11怎么关闭透明效果_Windows11辅助功能视觉效果设置
简单实现jsp分页
Java垃圾回收器的方法和原理总结
Laravel如何处理异常和错误?(Handler示例)
在Oracle关闭情况下如何修改spfile的参数
Laravel怎么在Blade中安全地输出原始HTML内容
如何在万网主机上快速搭建网站?
Laravel如何配置和使用缓存?(Redis代码示例)
Windows11怎样设置电源计划_Windows11电源计划调整攻略【指南】
laravel怎么在请求结束后执行任务(Terminable Middleware)_laravel Terminable Middleware请求结束任务执行方法
laravel怎么配置和使用PHP-FPM来优化性能_laravel PHP-FPM配置与性能优化方法
logo在线制作免费网站在线制作好吗,DW网页制作时,如何在网页标题前加上logo?
Win10如何卸载预装Edge扩展_Win10卸载Edge扩展教程【方法】
如何在阿里云通过域名搭建网站?
佛山企业网站制作公司有哪些,沟通100网上服务官网?
Laravel如何集成第三方登录_Laravel Socialite实现微信QQ微博登录
如何快速完成中国万网建站详细流程?
网站广告牌制作方法,街上的广告牌,横幅,用PS还是其他软件做的?
Laravel Admin后台管理框架推荐_Laravel快速开发后台工具
如何快速搭建支持数据库操作的智能建站平台?
Laravel如何实现多级无限分类_Laravel递归模型关联与树状数据输出【方法】
微信公众帐号开发教程之图文消息全攻略
如何在不使用负向后查找的情况下匹配特定条件前的换行符
邀请函制作网站有哪些,有没有做年会邀请函的网站啊?在线制作,模板很多的那种?
JS中使用new Date(str)创建时间对象不兼容firefox和ie的解决方法(两种)
黑客如何通过漏洞一步步攻陷网站服务器?
Python文件流缓冲机制_IO性能解析【教程】
Laravel Sail是什么_基于Docker的Laravel本地开发环境Sail入门
Python文本处理实践_日志清洗解析【指导】
Laravel怎么集成Vue.js_Laravel Mix配置Vue开发环境
Laravel DB事务怎么使用_Laravel数据库事务回滚操作
千问怎样用提示词获取健康建议_千问健康类提示词注意事项【指南】
Laravel怎么连接多个数据库_Laravel多数据库连接配置
详解Android中Activity的四大启动模式实验简述
php json中文编码为null的解决办法
Java遍历集合的三种方式
海南网站制作公司有哪些,海口网是哪家的?
Laravel如何实现多语言支持_Laravel本地化与国际化(i18n)配置教程
车管所网站制作流程,交警当场开简易程序处罚决定书,在交警网站查询不到怎么办?
如何快速查询域名建站关键信息?
Laravel如何配置中间件Middleware_Laravel自定义中间件拦截请求与权限校验【步骤】
Laravel API路由如何设计_Laravel构建RESTful API的路由最佳实践
Laravel中DTO是什么概念_在Laravel项目中使用数据传输对象(DTO)
Laravel如何设置定时任务(Cron Job)_Laravel调度器与任务计划配置
Laravel与Inertia.js怎么结合_使用Laravel和Inertia构建现代单页应用
如何在 Pandas 中基于一列条件计算另一列的分组均值
上一篇: 巩义网页设计,河南农产品品牌有哪些?
下一篇:java 反射机制
上一篇: 巩义网页设计,河南农产品品牌有哪些?
下一篇:java 反射机制


引与容量。