Java行为型模式---备忘录模式

备忘录模式基础概念

备忘录模式(Memento Pattern)是一种行为型设计模式,其核心思想是在不破坏封装性的前提下,捕获一个对象的内部状态,并在该对象之外保存这个状态,以便后续可以将该对象恢复到先前保存的状态。备忘录模式通过将状态保存和恢复的责任分离,实现了对象状态管理的封装和解耦。

备忘录模式的核心组件

  1. 原发器(Originator) - 需要保存状态的对象,负责创建和恢复备忘录。
  2. 备忘录(Memento) - 存储原发器的内部状态,通常通过原发器创建并由管理者持有。
  3. 管理者(Caretaker) - 负责保存备忘录,但不能对备忘录的内容进行操作或检查。

备忘录模式的实现

下面通过一个文本编辑器的例子展示备忘录模式的实现:

// 1. 备忘录类 - 存储原发器的状态
class Memento {private final String content;  // 需要保存的状态public Memento(String content) {this.content = content;}public String getContent() {return content;}
}// 2. 原发器 - 文本编辑器
class TextEditor {private String content;  // 内部状态public void setContent(String content) {this.content = content;}public String getContent() {return content;}// 创建备忘录,保存当前状态public Memento createMemento() {return new Memento(content);}// 从备忘录恢复状态public void restoreMemento(Memento memento) {this.content = memento.getContent();}
}// 3. 管理者 - 负责保存备忘录
class History {private final Stack<Memento> mementos = new Stack<>();  // 使用栈保存历史状态public void push(Memento memento) {mementos.push(memento);}public Memento pop() {if (mementos.isEmpty()) {return null;}return mementos.pop();}
}// 4. 客户端代码
public class MementoPatternClient {public static void main(String[] args) {TextEditor editor = new TextEditor();History history = new History();// 编辑文本并保存状态editor.setContent("Hello");history.push(editor.createMemento());editor.setContent("Hello World");history.push(editor.createMemento());editor.setContent("Hello World!");System.out.println("当前内容: " + editor.getContent());  // 输出: Hello World!// 撤销操作editor.restoreMemento(history.pop());System.out.println("撤销一次后的内容: " + editor.getContent());  // 输出: Hello Worldeditor.restoreMemento(history.pop());System.out.println("再撤销一次后的内容: " + editor.getContent());  // 输出: Hello}
}

备忘录模式的变体

  1. 白箱备忘录 - 备忘录的状态对所有类可见,违反封装原则,但实现简单:

    class Memento {private String content;public Memento(String content) {this.content = content;}// 所有类可访问的公共方法public String getContent() {return content;}public void setContent(String content) {this.content = content;}
    }
    
  2. 黑箱备忘录 - 使用内部类和接口实现封装,原发器以外的类无法访问备忘录内容:

    interface MementoInterface {// 空接口,仅用于标识
    }class TextEditor {private String content;// 私有内部类实现备忘录private class EditorMemento implements MementoInterface {private final String savedContent;public EditorMemento(String content) {this.savedContent = content;}private String getSavedContent() {return savedContent;}}public MementoInterface createMemento() {return new EditorMemento(content);}public void restoreMemento(MementoInterface memento) {EditorMemento editorMemento = (EditorMemento) memento;this.content = editorMemento.getSavedContent();}
    }
    
  3. 增量备忘录 - 只保存状态变化的部分,节省内存:

    class IncrementalMemento {private final String addedText;private final int position;public IncrementalMemento(String addedText, int position) {this.addedText = addedText;this.position = position;}// 恢复方法由原发器实现
    }
    

备忘录模式的应用场景

  1. 撤销 / 重做功能 - 如文本编辑器、图形设计工具的历史记录
  2. 事务管理 - 数据库操作的回滚机制
  3. 游戏存档 - 保存游戏进度,支持读取存档
  4. 状态恢复 - 系统崩溃后的恢复点机制
  5. 多级筛选 - 如电商平台的筛选条件保存与恢复
  6. 浏览器历史 - 网页浏览历史的前进 / 后退功能

备忘录模式的优缺点

优点

  • 保持封装性 - 不破坏对象的封装前提下保存和恢复其内部状态
  • 简化原发器 - 原发器不需要管理自己的历史状态,职责更清晰
  • 支持撤销操作 - 可以方便地实现多级撤销和重做功能
  • 状态恢复透明 - 客户端无需关心状态的保存细节
  • 符合开闭原则 - 可以在不修改原发器的情况下新增备忘录类

缺点

  • 资源消耗 - 如果频繁创建备忘录,会占用大量内存
  • 性能问题 - 对于大型对象,保存和恢复状态可能影响性能
  • 管理复杂度 - 管理者需要正确管理备忘录的生命周期
  • 序列化开销 - 如果需要跨进程或持久化保存,可能增加序列化复杂度

使用备忘录模式的注意事项

  1. 控制备忘录大小 - 避免保存过大的对象状态,考虑使用增量备忘录
  2. 限制管理者权限 - 管理者只能保存和传递备忘录,不能修改其内容
  3. 考虑内存管理 - 对于长期保存的备忘录,需要考虑内存回收策略
  4. 处理复杂对象 - 对于包含引用的对象,需要处理深拷贝和浅拷贝问题
  5. 结合其他模式 - 备忘录模式常与命令模式结合实现撤销系统,与迭代器模式结合遍历历史状态
  6. 持久化支持 - 如果需要持久化备忘录,需考虑序列化和版本兼容性

总结

备忘录模式通过将对象状态的保存和恢复封装在独立的备忘录类中,实现了对象状态管理的解耦和封装。它在不破坏对象封装性的前提下,允许对象在需要时恢复到之前的状态,是实现撤销 / 重做功能、事务回滚、游戏存档等场景的理想选择。在实际开发中,合理使用备忘录模式可以提高系统的可维护性和用户体验,但需要注意控制内存消耗和管理复杂度。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如若转载,请注明出处:http://www.pswp.cn/bicheng/89930.shtml
繁体地址,请注明出处:http://hk.pswp.cn/bicheng/89930.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

后端参数校验

前端给后端传输数据&#xff0c;有时候参数需要校验&#xff0c;我们自己写代码会比较麻烦&#xff0c;我们可以使用springboot为我们提供的注解&#xff0c;降低这些没有必要的代码开发。1.引入依赖<dependency><groupId>org.springframework.boot</groupId>…

C++ - 仿 RabbitMQ 实现消息队列--服务端核心模块实现(一)

目录 日志打印工具 实用 Helper 工具 sqlite 基础操作类 字符串操作类 UUID 生成器类 文件基础操作 文件是否存在判断 文件大小获取 读文件 写文件 文件重命名 文件创建/删除 父级目录的获取 目录创建/删除 附录&#xff08;完整代码&#xff09; 日志打印工具 为了便…

C语言:第07天笔记

C语言&#xff1a;第07天笔记 内容提要 循环结构 break与continue 综合案例《猜拳游戏》数组 数组的概念一维数组流程控制 break与continue break 功能&#xff1a; ① 用在switch中&#xff0c;用来跳出switch中的case语句&#xff1b;如果case没有break&#xff0c;可能会产生…

qt 中英文翻译 如何配置和使用

qt 中英文翻译 如何配置和使用 1. 在.pro文件中添加TRANSLATIONS 在你的 .pro 文件&#xff08;比如 HYAC_AAF_HOST.pro&#xff09;中添加&#xff1a; TRANSLATIONS \ zh\_CN.ts en\_US.ts这会告诉Qt项目你要支持中文和英文。 2. 提取可翻译文本&#xff08;生成ts文件&#…

Leetcode 710. 黑名单中的随机数

1.题目基本信息 1.1.题目描述 给定一个整数 n 和一个 无重复 黑名单整数数组 blacklist 。设计一种算法&#xff0c;从 [0, n - 1] 范围内的任意整数中选取一个 未加入 黑名单 blacklist 的整数。任何在上述范围内且不在黑名单 blacklist 中的整数都应该有 同等的可能性 被返…

RxJava 全解析:从原理到 Android 实战

在 Android 开发中&#xff0c;异步任务处理是绕不开的核心场景 —— 网络请求、数据库操作、文件读写等都需要在后台执行&#xff0c;而结果需回调到主线程更新 UI。传统的 “HandlerThread” 或 AsyncTask 不仅代码冗余&#xff0c;还容易陷入 “回调地狱”&#xff08;嵌套回…

OpenCV 官翻7 - 对象检测

文章目录ArUco 标记检测标记与字典标记物创建标记检测姿态估计选择字典预定义字典自动生成字典手动定义字典检测器参数阈值处理adaptiveThreshConstant轮廓过滤minMarkerPerimeterRate 与 maxMarkerPerimeterRatepolygonalApproxAccuracyRateminCornerDistanceRateminMarkerDis…

【Oracle】ORACLE OMF说明

ORACLE OMF (Oracle Managed Files) 是 Oracle 数据库提供的一项自动化文件管理功能。它的核心目的是简化数据库管理员&#xff08;DBA&#xff09;对数据库底层操作系统文件的管理工作。 以下是 OMF 的关键要点&#xff1a; 核心功能&#xff1a;自动命名和定位文件 在创建数据…

408考研逐题详解:2010年第35题——RIP协议

2010年第35题 某自治系统内采用 RIP 协议&#xff0c;若该自治系统内的路由器 R1 收到其邻居路由器 R2 的距离矢量&#xff0c;距离矢量中包含信息 <net1, 16>&#xff0c;则能得出的结论是&#xff08; &#xff09; A. R2 可以经过 R1 到达 net1&#xff0c;跳数为17 …

http与https的主要区别是什么?

什么是HTTP&#xff1f; HTTP&#xff08;HyperText Transfer Protocol&#xff0c;超文本传输协议&#xff09;是互联网上应用最为广泛的一种网络协议。它构成了Web数据通信的基础&#xff0c;并定义了客户端和服务器之间如何请求和传递网页信息。当您在浏览器中输入一个网址时…

STC89C52系列单片机简介

STC89C52系列单片机是由中国宏晶科技&#xff08;STC&#xff09;推出的一款新一代增强型8051内核单片机。它不仅继承了传统8051指令系统的兼容性&#xff0c;还在性能、功耗、抗干扰能力以及性价比方面进行了全面提升&#xff0c;广泛应用于各类嵌入式控制场景&#xff0c;如工…

基于 Docker 环境的 JupyterHub 详细部署手册

本文详细介绍基于Docker Compose的单机版JupyterHub部署方案&#xff0c;通过容器化技术实现多用户Notebook环境的快速搭建。方案采用官方JupyterHub镜像&#xff0c;配置11个端口映射&#xff08;18000-18010&#xff09;支持用户并发&#xff0c;通过数据卷挂载&#xff08;.…

常见的万能密码

目录 1. 通用SQL注入 2. 登录绕过 3. 密码重置 1. 通用SQL注入 or 11-- " or 11-- or aa " or "a""a or 11# " or 11# or 11/* " or 11/* or 11 " or "1""1 2. 登录绕过 admin-- admin or 11-- admin or aa …

04训练windows电脑低算力显卡如何部署pytorch实现GPU加速

大多数人用的电脑的显卡配置可能没有那么强,也就是说,我们很难享受到最新版本pytorch给我们带来的模型训练的速度和效率,为此,我们需要想办法在现有显卡情况下部署应用pytorch。 笔者有一台电脑,显卡算力很低,那么以该电脑为例,为大家介绍如何部署应用pytorch功能。 1…

PPT科研画图插件

PPT科研画图插件 iSlide- 让PPT设计简单起来 | PPT模板下载平台iSlide - 简单&#xff0c;好用的PPT插件&#xff01;拥有30万 原创可商用PPT模板&#xff0c;PPT主题素材&#xff0c;PPT案例&#xff0c;PPT图表&#xff0c;PPT图示&#xff0c;PPT图标&#xff0c;PPT插图和8…

CSS实现背景图片渐变透明

复合写法background: linear-gradient(180deg, rgba(255, 255, 255, 0) 0%, #FFF 82.5%),url(https://example.com/image.jpg) center / cover no-repeat;参数说明&#xff1a;linear-gradient(180deg, rgba(255, 255, 255, 0) 0%, #FFF 82.5%)创建从下至上的垂直渐变&#xff…

基于pyside6的通用机器人遥控控制界面

1. 前言 这两天需要帮一个朋友做一个简单的遥控控制界面&#xff0c;用于控制一台复合机器人(万向轮底盘机械臂旋转云台)&#xff0c;在这里分享一下 2. 开发框架 由于朋友那边的控制接口都是使用python来写的&#xff0c;所以我这里也使用py来完成这个遥控界面的开发。但其…

【iOS】ZARA仿写

【iOS】ZARA仿写 文章目录【iOS】ZARA仿写前言首页发现我的对姓名的更改总结前言 暑假第一个的任务仿写ZARA 虽然不是特别难却有很多小细节需要注意 首页 点进程序出现的就是整个项目最主要的一个点&#xff0c;即首页的无限轮播图&#xff0c;不管是自动轮播还是手动滑动&a…

Kubernetes Pod 调度基础

一、Replication Controller 和 ReplicaSet1、Replication ControllerReplication Controller&#xff08;复制控制器&#xff0c;RC&#xff09;RC 用来确保 Pod 副本数达到期望值&#xff0c;这样可以确保一个或多个同类 Pod 总是可用的。如果存在的 Pod 数量大于设定的值&am…

菜鸟的C#学习(二)

文章目录一、类的访问1、普通类继承抽象类2、普通类继承抽象类&#xff0c;抽象类继承接口&#xff0c;三者联系二、类中方法的访问2.1 抽象方法和虚方法2.2 虚方法和普通方法**1. 调用机制****2. 方法重写****3. 设计意图****4. 性能差异****5. 语法对比表****总结&#xff1a…