Go中备忘录模式如何保存状态_Go备忘录模式数据回滚实现
发布时间 - 2026-01-30 00:00:00 点击率:次Go中备忘录模式的核心难点是确保仅Originator可读写Memento而Caretaker只能持有:需用全导出字段的不可变结构体,深拷贝复杂状态,限制历史快照数量防内存泄漏,并避免存储临时资源句柄。
备忘录模式在 Go 中的核心实现难点
Go 没有内置的类、继承或访问控制(如 private),所以无法像 Java/C# 那样靠封装强制隔离备忘录内部状态。真正的难点不是“怎么存”,而是“怎么确保只有原发起者(Originator)能读写备忘录内容,而管理者(Caretaker)只能持有、不能篡改”。否则回滚就失去意义。
Memento 必须是不可变结构体 + 原始字段暴露
常见错误是把 Memento 设计成带方法或私有字段的类型,结果 Caretaker 无法序列化、无法深拷贝、甚至无法安全传递。正确做法是让它成为纯数据载体:
type Memento struct {
State string
Version int
Timestamp int64
}
关键点:
-
Memento字段全部导出(首字母大写),否则外部包(包括 Caretaker)无法读取 - 不提供任何 setter 方法,也不嵌套指针或 map/slice 等可变引用——避免外部修改影响原始快照
- 如果需保存复杂状态,用
json.Raw或
Message
[]byte存序列化结果,而不是直接存 struct 指针
Originator 的 Save() 和 Restore() 必须严格配对
回滚失效最常见的原因是状态复制不完整。比如 Originator 内部用了 map 或切片,Save() 只浅拷贝了引用:
func (o *Originator) Save() *Memento {
return &Memento{
State: o.state, // ✅ string 是值类型,安全
Data: o.cache, // ❌ 若 cache 是 map[string]int,这里只是引用!
}
}
修复方式:
- 对 slice:用
append([]T(nil), src...)深拷贝 - 对 map:手动遍历重建新 map
- 更稳妥的做法是统一走 JSON 编解码:
json.Marshal→Memento.Payload []byte→json.Unmarshal -
Restore()必须完全覆盖当前字段,不要做“merge”或“partial update”
Caretaker 用 slice 管理历史时要注意内存泄漏
很多示例直接用 []*Memento 存所有快照,但没限制长度,长期运行后 OOM。实际使用中必须加约束:
- 设置最大保存数量(如最多 50 个),超出时用
append(history[1:], m)截断旧记录 - 若
Memento含大字段(如原始图片字节),考虑只存 diff 或用 LRU 缓存淘汰策略 - 回滚操作后,建议显式将已废弃的
Memento置为nil(虽不能强制 GC,但能减少误用)
真正容易被忽略的是:Go 中没有析构函数,一旦 Memento 被 Caretaker 持有,它的生命周期就脱离 Originator 控制——所以设计上要默认它会被长期持有,别在里面塞临时资源句柄(如 *os.File)。
# java
# js
# json
# go
# app
# 字节
# c#
# golang
# 封装
# 析构函数
# 结构体
# 指针
# 继承
# private
# Struct
# 切片
# nil
# append
# map
# history
# 句柄
# 的是
# 也不
# 序列化
# 最多
# 遍历
# 用了
# 要做
# 在里面
# 让它
相关栏目:
【
网站优化151355 】
【
网络推广146373 】
【
网络技术251813 】
【
AI营销90571 】
相关推荐:
Laravel如何配置任务调度?(Cron Job示例)
网页设计与网站制作内容,怎样注册网站?
JS中页面与页面之间超链接跳转中文乱码问题的解决办法
如何确保西部建站助手FTP传输的安全性?
惠州网站建设制作推广,惠州市华视达文化传媒有限公司怎么样?
LinuxShell函数封装方法_脚本复用设计思路【教程】
Laravel如何实现全文搜索功能?(Scout和Algolia示例)
中山网站推广排名,中山信息港登录入口?
高端智能建站公司优选:品牌定制与SEO优化一站式服务
齐河建站公司:营销型网站建设与SEO优化双核驱动策略
Laravel如何使用withoutEvents方法临时禁用模型事件
Laravel如何优化应用性能?(缓存和优化命令)
laravel服务容器和依赖注入怎么理解_laravel服务容器与依赖注入解析
在centOS 7安装mysql 5.7的详细教程
Laravel Session怎么存储_Laravel Session驱动配置详解
Microsoft Edge如何解决网页加载问题 Edge浏览器加载问题修复
Laravel如何实现多表关联模型定义_Laravel多对多关系及中间表数据存取【方法】
Laravel怎么做数据加密_Laravel内置Crypt门面的加密与解密功能
Laravel怎么实现模型属性转换Casting_Laravel自动将JSON字段转为数组【技巧】
html5的keygen标签为什么废弃_替代方案说明【解答】
邀请函制作网站有哪些,有没有做年会邀请函的网站啊?在线制作,模板很多的那种?
Python正则表达式进阶教程_复杂匹配与分组替换解析
如何在Windows服务器上快速搭建网站?
如何快速搭建高效WAP手机网站?
node.js报错:Cannot find module 'ejs'的解决办法
Swift中swift中的switch 语句
打开php文件提示内存不足_怎么调整php内存限制【解决方案】
专业企业网站设计制作公司,如何理解商贸企业的统一配送和分销网络建设?
重庆市网站制作公司,重庆招聘网站哪个好?
实例解析angularjs的filter过滤器
HTML5建模怎么导出为FBX格式_FBX格式兼容性及导出步骤【指南】
微信小程序 五星评分(包括半颗星评分)实例代码
如何登录建站主机?访问步骤全解析
音乐网站服务器如何优化API响应速度?
利用python获取某年中每个月的第一天和最后一天
如何在Windows环境下新建FTP站点并设置权限?
Laravel Octane如何提升性能_使用Laravel Octane加速你的应用
php json中文编码为null的解决办法
HTML5空格和margin有啥区别_空格与外边距的使用场景【说明】
桂林网站制作公司有哪些,桂林马拉松怎么报名?
浅析上传头像示例及其注意事项
三星网站视频制作教程下载,三星w23网页如何全屏?
Laravel如何使用Gate和Policy进行授权?(权限控制)
javascript中的try catch异常捕获机制用法分析
电视网站制作tvbox接口,云海电视怎样自定义添加电视源?
VIVO手机上del键无效OnKeyListener不响应的原因及解决方法
如何在 Python 中将列表项按字母顺序编号(a.、b.、c. …)
Angular 表单中正确绑定输入值以确保提交与验证正常工作
利用 Google AI 进行 YouTube 视频 SEO 描述优化
Laravel Artisan命令怎么自定义_创建自己的Laravel命令行工具完全指南


