当前位置: 首页 > news >正文

设计模式学习(21) 23-19 备忘录模式

文章目录

  • 0. 个人感悟
  • 1. 概念
  • 2. 适配场景
    • 2.1 适合的场景
    • 2.2 常见场景举例
  • 3. 实现方法
    • 3.1 实现思路
    • 3.2 UML类图
    • 3.3 代码示例
  • 4. 优缺点
    • 4.1 优点
    • 4.2 缺点
  • 5. 源码分析
    • 5.1 Java Swing中的UndoManager

0. 个人感悟

  • 备忘录模式的场景也比较专。适合进行备份、恢复
  • 模式优点很明显,状态保存于业务逻辑分离,实现解耦
  • 一个注意的点是窄接口设计,Memento(备忘录)内部保存了Originator(原发器)的状态,但这个状态不应该被除了Originator之外的其他对象(特别是Caretaker)直接访问或修改,以保护状态的封装性。具体可以看看示例代码

1. 概念

英文定义(《设计模式:可复用面向对象软件的基础》)

Without violating encapsulation, capture and externalize an object,t internal state so that the object can be restored this state later.

中文翻译

在不破坏封装性的前提下,捕获一个对象的内部状态,并在该对象之外保存这个状态,以便以后可以将对象恢复到原先保存的状态。

理解

  • 备忘录模式提供了一种状态恢复机制,允许对象回到某个历史状态
  • 它通过独立的对象(备忘录)来存储状态,而不是由原对象自己保存
  • 实现了状态保存与业务逻辑的分离,原对象专注于业务,管理者专注于状态管理

2. 适配场景

2.1 适合的场景

  1. 需要撤销/重做功能:如文本编辑器、绘图软件的操作历史
  2. 需要保存快照/检查点:如游戏存档、系统配置备份
  3. 状态转换复杂:如工作流引擎的状态管理
  4. 需要事务回滚:如数据库操作的事务管理
  5. 原型状态保存:如复杂对象构造过程中的中间状态保存
  6. 需要版本控制:如文档编辑的历史版本管理

2.2 常见场景举例

  1. IDE开发环境:Eclipse/IntelliJ IDEA的代码编辑撤销功能
  2. 绘图软件:Photoshop的历史记录面板
  3. 游戏开发:游戏进度存档/读档系统
  4. 表单应用:多步表单的回退功能
  5. 配置管理:系统配置的备份与恢复
  6. 数据库系统:事务的rollback机制

3. 实现方法

3.1 实现思路

  1. 识别状态数据:确定需要保存的内部状态字段
  2. 创建Memento类:设计存储状态的数据结构
  3. 在Originator中添加状态管理
    • 创建保存状态的方法(saveToMemento()
    • 创建恢复状态的方法(restoreFromMemento()
  4. 设计Caretaker类
    • 决定存储策略(栈、列表、树等)
    • 实现状态管理逻辑(撤销、重做、清空等)
  5. 控制访问权限
    • 对Originator开放宽接口(可读写状态)
    • 对Caretaker开放窄接口(只读元数据)

3.2 UML类图

角色说明:

  • Originator(原发器):需要保存状态的对象,负责创建和恢复备忘录
  • Memento(备忘录):存储Originator内部状态,通常设计为不可变对象
  • Caretaker(管理者):负责保存和管理备忘录,但不能操作备忘录内容

3.3 代码示例

背景
以文件编辑器为例,简化业务,只有保存和撤销功能
设计

原发器

publicclassTextEditor{privateStringcontent;publicStringgetContent(){returncontent;}publicvoidsetContent(Stringcontent){this.content=content;}publicTextEditor(Stringcontent){this.content=content;}/** * @description 创建备忘 * @author bigHao * @date 2026/1/27 * @return designpattern.memento.pac.TextMemento 备忘录 **/publicTextMementocreateMemento(){returnnewTextMemento(content);}/** * @description 恢复备忘 * @author bigHao * @date 2026/1/27 * @param memento 备忘录 **/publicvoidrestoreMemento(TextMementomemento){this.content=memento.getContent();}}

备忘录

  • 注意权限控制。备忘录只有原发器可以创建和访问内容。实际操作中将二者放到一个包下,相关方法使用包权限
publicclassTextMemento{privateStringcontent;// 包访问权限,仅对Originator开放状态访问TextMemento(Stringcontent){this.content=content;}StringgetContent(){returncontent;}}

管理者

publicclassCaretaker{privatestaticfinalintMAX_SIZE=100;// 业务场景刚好使用stack 先进后出privateStack<TextMemento>history=newStack<>();publicvoidpush(TextMementomemento){// 限制下长度limitHistory();this.history.push(memento);}publicTextMementopop(){returnthis.history.pop();}privatevoidlimitHistory(){if(history.size()>MAX_SIZE){history.removeFirst();}}}

测试

publicclassClient{staticvoidmain(){// 创建Stringcontent="备忘录模式笔记";TextEditortextEditor=newTextEditor(content);printInfo(textEditor);Caretakercaretaker=newCaretaker();// 安全考虑,TextMemento是包访问权限,和TextEditor同包,其它地方无法直接创建// new TextMemento(""); 会报错// 保存1caretaker.push(textEditor.createMemento());content+="包括三个角色,分别是原发器";textEditor.setContent(content);caretaker.push(textEditor.createMemento());printInfo(textEditor);// 保存2content+="、备忘录";textEditor.setContent(content);caretaker.push(textEditor.createMemento());printInfo(textEditor);// 保存3content+="和管理者";textEditor.setContent(content);caretaker.push(textEditor.createMemento());printInfo(textEditor);// 编辑content+=" 功能分别是";textEditor.setContent(content);printInfo(textEditor);// 撤销1System.out.println("===撤销===");textEditor.restoreMemento(caretaker.pop());printInfo(textEditor);// 撤销2System.out.println("===撤销===");textEditor.restoreMemento(caretaker.pop());printInfo(textEditor);}privatestaticvoidprintInfo(TextEditortextEditor){System.out.println("当前内容: "+textEditor.getContent());}}

输出

当前内容: 备忘录模式笔记 当前内容: 备忘录模式笔记包括三个角色,分别是原发器 当前内容: 备忘录模式笔记包括三个角色,分别是原发器、备忘录 当前内容: 备忘录模式笔记包括三个角色,分别是原发器、备忘录和管理者 当前内容: 备忘录模式笔记包括三个角色,分别是原发器、备忘录和管理者 功能分别是 ===撤销=== 当前内容: 备忘录模式笔记包括三个角色,分别是原发器、备忘录和管理者 ===撤销=== 当前内容: 备忘录模式笔记包括三个角色,分别是原发器、备忘录

4. 优缺点

4.1 优点

高内聚低耦合

  • 职责分离:状态保存与业务逻辑解耦,Originator只关注核心功能
  • 封装性:通过窄接口模式保护状态不被Caretaker误操作

复用性

  • 状态管理可复用:Caretaker可被多个Originator复用
  • 存储策略灵活:支持栈、列表、数据库等多种存储方式

可读性

  • 意图明确:代码明确表达了"保存状态"和"恢复状态"的意图
  • 结构清晰:三个角色分工明确,易于理解

维护性

  • 易于扩展:可轻松添加新的存储策略或状态类型
  • 修改隔离:状态存储逻辑变化不影响Originator

稳定性

  • 状态一致:确保状态恢复的正确性
  • 异常安全:状态操作失败时可恢复到之前状态

4.2 缺点

资源消耗

  • 内存占用:大量状态保存可能导致内存消耗过大
  • 性能开销:频繁的状态保存/恢复可能影响性能

设计复杂度

  • 额外类:增加了Memento和Caretaker类
  • 窄接口实现复杂:需要小心设计访问权限

状态同步

  • 深拷贝问题:复杂对象的深拷贝可能实现复杂
  • 版本兼容:状态数据结构变化时,旧备忘录可能无法恢复

5. 源码分析

5.1 Java Swing中的UndoManager

代码位置javax.swing.undo.UndoManager

角色分析

// Originator:各种Document类(如PlainDocument)publicclassPlainDocumentextendsAbstractDocument{// 创建UndoableEdit(备忘录)protectedUndoableEditcreateUndoableEdit(ElementChangeec){returnnewDocumentUndoableEdit(ec);}}// Memento:UndoableEdit接口及其实现publicinterfaceUndoableEdit{voidundo()throwsCannotUndoException;voidredo()throwsCannotRedoException;booleancanUndo();booleancanRedo();}// Caretaker:UndoManager类publicclassUndoManagerextendsCompoundEditimplementsUndoableEditListener{privateVector<UndoableEdit>edits;// 存储备忘录privateintindexOfNextAdd;// 下一个添加位置publicvoidundo()throwsCannotUndoException{// 执行撤销操作UndoableEditedit=edits.elementAt(--indexOfNextAdd);edit.undo();}publicvoidredo()throwsCannotRedoException{// 执行重做操作UndoableEditedit=edits.elementAt(indexOfNextAdd);edit.redo();indexOfNextAdd++;}}

参考:

  • 韩顺平 Java设计模式
  • 归思君 设计模式学习笔记(十八)备忘录模式及其实现
  • kosamino 设计模式之备忘录模式(Memento)详解及代码示例
http://www.jsqmd.com/news/307648/

相关文章:

  • d3dcompiler_47.dll缺失怎么修复 免费下载文件方法分享
  • 【Django毕设全套源码+文档】基于Django框架的贵州特色产品销售平台的设计与实现(丰富项目+远程调试+讲解+定制)
  • VSCode 附加进程调试完整指南
  • 2026年中国留学生求职机构推荐:行业应用深度评测直击适配与内推关键痛点
  • 2026年行业内优秀的冷却塔品牌怎么选,玻璃钢冷却塔/冷却塔/工业冷却塔/圆形逆流冷却塔,冷却塔制造厂家排行榜
  • 实验台哪家强?五大口碑品牌深度解析,总有一款适配你的科研需求
  • 五恒系统厂商推荐,性价比高的公司有哪些?
  • 2026年美国求职机构推荐:留学生求职市场排名,涵盖科技金融行业与OPT痛点
  • VSCode,Claude 插件使用第三方 API 总是提示登录?
  • 聊聊新中式铝艺影壁墙服务商怎么选,哪家值得推荐
  • 东岳搬家联系方式:评估搬家服务前的实用信息
  • 定制门窗一线品牌贝克洛多少钱,价格合理吗
  • 2026年官厅中华永久陵园电话推荐:基于场景与合规评测,解决信息查询与交通痛点
  • 分析皖南电机定制专家,性价比高的是哪家
  • 2026年美国求职机构推荐:聚焦科技金融行业评价,针对OPT与H1B核心痛点指南
  • 涨跌停股票实时数据API接口汇总
  • 全国雅思培训排行推荐:2026权威出国雅思课程中心学校口碑排行榜
  • 大语言模型演进史丨智能涌现之后,路在何方?(上)
  • 全国雅思培训排行推荐-2026权威出国雅思课程中心学校口碑排行榜
  • 解析CI/CD工具TeamCity的可视化失败追踪功能
  • 吐血推荐8个一键生成论文工具,自考毕业论文轻松搞定!
  • 非参数检验,又称分布自由检验,是一类不依赖于特定分布形式的统计检验方法
  • 分析烟囱防腐来样定制,价格合理的公司怎么选择
  • Wilcoxon秩和检验
  • 2026年成都高性价比点焊机生产厂排名,台式、次级整流点焊机厂家盘点
  • 全国雅思培训排行推荐,2026权威出国雅思课程中心学校口碑排行榜
  • 2026年上海装修装饰设计公司排名,朗域装饰凭高性价比上榜推荐
  • 中山的锴锋机械品牌口碑如何,其团队凝聚力好不好?
  • 聊聊1.8W贴片太阳能板定制,费用和品牌怎么选
  • Pipeline:Beam如何抽象多步骤的数据流水线?