备忘录模式:实现对象状态撤销与恢复的设计模式
备忘录模式:实现对象状态撤销与恢复的设计模式
一、模式核心:在不破坏封装性的前提下保存和恢复对象状态
在软件开发中,经常需要实现 “撤销” 功能(如文本编辑器的撤销修改、游戏存档读取)。直接暴露对象内部状态会破坏封装性,而备忘录模式通过独立的备忘录对象封装状态,实现安全的状态管理。
备忘录模式(Memento Pattern) 允许在不暴露对象内部细节的情况下,捕获对象的内部状态并保存为备忘录(Memento),后续可通过备忘录恢复对象状态。核心解决:
- 状态封装:备忘录对象封装对象状态,避免外部直接访问内部属性。
- 撤销 / 重做支持:通过保存多个备忘录实现多步撤销(如版本控制)。
- 单一职责分离:将状态管理逻辑从原发对象中分离,符合开闭原则。
核心思想与 UML 类图(PlantUML 语法)
备忘录模式包含以下角色:
- 原发器(Originator):创建并恢复自身状态的对象。
- 备忘录(Memento):存储原发器的状态,提供有限访问接口。
- 管理者(Caretaker):管理备忘录的创建、存储和获取(如历史记录列表)。
二、核心实现:文本编辑器的撤销功能
1. 定义原发器(文本编辑器)
public class TextEditor { private String content; // 编辑内容 // 创建备忘录(保存当前状态) public Memento createMemento() { return new Memento(content); } // 恢复状态(从备忘录中读取) public void restoreMemento(Memento memento) { this.content = memento.getState(); } // 修改内容(模拟编辑操作) public void append(String text) { content = (content != null ? content : "") + text; } // 获取当前内容 public String getContent() { return content; }
}
2. 定义备忘录(包可见,隐藏状态访问)
class Memento { private final String state; Memento(String state) { this.state = state; } // 包可见方法,仅同一包内的原发器可调用 String getState() { return state; }
}
3. 定义管理者(保存历史记录)
import java.util.ArrayList;
import java.util.List; public class HistoryManager { private final List<Memento> mementoList = new ArrayList<>(); // 添加新备忘录到历史记录 public void saveMemento(Memento memento) { mementoList.add(memento); } // 获取指定版本的备忘录(索引从 0 开始) public Memento getMemento(int index) { return mementoList.get(index); }
}
4. 客户端使用备忘录模式
public class ClientDemo { public static void main(String[] args) { TextEditor editor = new TextEditor(); HistoryManager history = new HistoryManager(); // 编辑步骤 1:输入 "Hello" editor.append("Hello"); System.out.println("当前内容:" + editor.getContent()); // 输出:Hello history.saveMemento(editor.createMemento()); // 保存状态 1 // 编辑步骤 2:添加 " World!" editor.append(" World!"); System.out.println("当前内容:" + editor.getContent()); // 输出:Hello World! history.saveMemento(editor.createMemento()); // 保存状态 2 // 撤销到第一步 editor.restoreMemento(history.getMemento(0)); System.out.println("撤销后内容:" + editor.getContent()); // 输出:Hello }
}
输出结果:
当前内容:Hello
当前内容:Hello World!
撤销后内容:Hello
三、进阶:实现多步撤销与状态快照
通过在管理者中维护备忘录列表,支持回滚到任意历史版本(如版本控制系统)。
1. 扩展管理者支持版本索引
public class AdvancedHistoryManager { private final List<Memento> mementoList = new ArrayList<>(); private int currentIndex = -1; // 当前版本索引 // 保存新状态并清除后续版本(如重做后新增修改) public void saveMemento(Memento memento) { currentIndex++; if (currentIndex < mementoList.size()) { mementoList.set(currentIndex, memento); // 覆盖旧版本 } else { mementoList.add(memento); // 添加新版本 } } // 撤销(回退到上一版本) public Memento undo() { if (currentIndex > 0) { currentIndex--; return mementoList.get(currentIndex); } return null; } // 重做(前进到下一版本) public Memento redo() { if (currentIndex < mementoList.size() - 1) { currentIndex++; return mementoList.get(currentIndex); } return null; }
}
2. 客户端测试多步撤销 / 重做
public class ClientDemo { public static void main(String[] args) { AdvancedHistoryManager history = new AdvancedHistoryManager(); TextEditor editor = new TextEditor(); // 编辑并保存三个版本 editor.append("A"); history.saveMemento(editor.createMemento()); // 版本 0: "A" editor.append("B"); history.saveMemento(editor.createMemento()); // 版本 1: "AB" editor.append("C"); history.saveMemento(editor.createMemento()); // 版本 2: "ABC" // 撤销两次到版本 0 history.undo(); // 版本 1 history.undo(); // 版本 0 System.out.println("撤销两次后:" + editor.getContent()); // 输出:"A" // 重做一次到版本 1 editor.restoreMemento(history.redo()); System.out.println("重做一次后:" + editor.getContent()); // 输出:"AB" }
}
四、框架与源码中的备忘录实践
1. Java 的 java.io.Serializable
对象序列化可视为备忘录模式的一种实现:通过序列化保存对象状态(备忘录),反序列化恢复状态(如分布式系统中的 checkpoint)。
2. Git 版本控制
Git 通过提交(Commit)保存代码快照(备忘录),允许回滚到任意历史版本(git checkout
/git revert
),本质上是备忘录模式的应用。
3. Eclipse 的撤销功能
Eclipse 编辑器通过备忘录模式保存代码修改历史,支持多步撤销(Ctrl+Z
)和重做(Ctrl+Y
),每个修改版本对应一个备忘录。
五、避坑指南:正确使用备忘录模式的 3 个要点
1. 控制备忘录的访问权限
备忘录的状态访问方法应设为包可见(默认权限)或私有,避免外部直接修改状态,确保仅原发器可恢复状态。
2. 处理大状态的性能问题
若对象状态包含大量数据(如图像、大文件),备忘录会占用大量内存。可采用原型模式克隆轻量级状态,或虚拟备忘录仅记录差异(如命令模式中的增量保存)。
3. 避免内存泄漏
管理者需合理管理备忘录列表,及时清理不再需要的历史记录(如限制最大版本数),防止内存溢出。
六、总结:何时该用备忘录模式?
适用场景 | 核心特征 | 典型案例 |
---|---|---|
撤销 / 重做功能 | 需要记录对象状态变化,支持回滚 | 文本编辑器、绘图软件 |
状态备份与恢复 | 定期保存状态快照,支持故障恢复 | 游戏存档、数据库备份 |
版本控制 | 需要管理对象的多个历史版本 | 代码版本管理、文档修订追踪 |
备忘录模式通过封装状态管理逻辑,在不破坏封装性的前提下实现了灵活的状态恢复机制。下一篇我们将探讨中介者模式,解析如何解耦对象间的复杂交互,敬请期待!
扩展思考:备忘录模式 vs 命令模式
类型 | 核心功能 | 协作方式 |
---|---|---|
备忘录模式 | 保存 / 恢复对象状态 | 原发器创建备忘录,管理者存储 |
命令模式 | 封装操作命令,支持撤销 / 重做 | 命令对象记录操作前后状态 |
组合使用 | 命令模式调用备忘录模式保存操作状态,实现多步撤销 | 命令执行时创建备忘录,撤销时恢复 |
相关文章:

备忘录模式:实现对象状态撤销与恢复的设计模式
备忘录模式:实现对象状态撤销与恢复的设计模式 一、模式核心:在不破坏封装性的前提下保存和恢复对象状态 在软件开发中,经常需要实现 “撤销” 功能(如文本编辑器的撤销修改、游戏存档读取)。直接暴露对象内部状态会…...
蓝桥杯 5. 交换瓶子
交换瓶子 原题目链接 题目描述 有 N 个瓶子,编号为 1 ~ N,放在架子上。 例如有 5 个瓶子,当前排列为: 2 1 3 5 4每次可以拿起 2 个瓶子,交换它们的位置。 要求通过若干次交换,使得瓶子的编号从小到大…...
本地使用Ollama部署DeepSeek
以下是在本地使用Ollama部署DeepSeek的详细教程,涵盖安装、修改安装目录、安装大模型以及删除大模型的操作步骤。 安装Ollama 1. 系统要求 确保你的系统满足以下条件: 操作系统:macOS、Linux或者Windows。足够的磁盘空间和内存。 2. 安装…...

freecad参数化三维模型装配体解析至web端,切换参数组或修改参数
用免费开源的freecad制作全参数化的三维模型,并且装配,上传至服务器,解析至web端,用户可以切换参数或修改参数,驱动模型改变。 freecad全参数化装配体模型解析至web端进行参数切换、修改完整展示_哔哩哔哩_bilibili …...
前端基础之《Vue(9)—混入》
一、什么是混入 1、是一种代码复用的技巧 Vue组件是由若干选项组成的,向组件中混入可复用的选项。 2、作用 比如我封装两个组件,一个是A组件,一个是B组件,发现它里面有相同的选项,就可以用混用的方式来复用它。 二、…...
ORACLE DATAGUARD遇到GAP增量恢复方式修复RAC环境备机的实践
ORACLE DATAGUARD技术是一个常用的数据保护机制,在DATAGUARD运行过程中,遇到异常导致备机不同步,而主库的归档日志也被清理,此时出现GAP,无法同步;就需要人工处理;对于小型数据库重新全量同步数…...
机器人进阶---视觉算法(六)傅里叶变换在图像处理中怎么用
傅里叶变换在图像处理中怎么用 傅里叶变换的基本原理应用场景Python代码示例逐行解释总结傅里叶变换在图像处理中是一种重要的工具,它将图像从空间域转换到频域,从而可以对图像的频率特性进行分析和处理。傅里叶变换在图像滤波、图像增强、图像压缩和图像分析等方面都有广泛应…...
Java知识日常巩固(五)
Java中wait()和 sleep()的区别? 在Java中,wait()和sleep()方法用于线程控制,但它们之间存在几个关键区别: 1. 用途 wait():用于线程间的协作。当一个线程需要等待某个条件满足时,它会调用wait()方法释放锁并进入等待状态,直到其他线程调用相同对象的notify()或notifyAl…...

浅析锁的应用与场景
锁的应用与场景:从单机到分布式 摘要:在多线程和分布式系统中,“锁”是避免资源竞争、保障数据一致性的核心机制。但你真的了解锁吗?什么时候该用锁?用哪种锁?本文通过通俗的比喻和代码示例,带…...

语音合成之五语音合成中的“一对多”问题主流模型解决方案分析
语音合成中的“一对多”问题主流模型解决方案分析 引言“一对多”指的是什么?优秀开源模型的方法CosyvoiceSparkTTSLlaSA TTSVITS 引言 TTS系统旨在模仿人类的自然语音,但其核心面临着一个固有的挑战,即“一对多”问题 。这意味着对于给定的…...

ElementUi的Dropdown下拉菜单的详细介绍及使用
Dropdown是 ElementUI 中用于创建下拉菜单项的一个组件,通常el-dropdown-item 包裹在 el-dropdown 组件中使用。以下从功能特性(一些属性及方法)、使用和高级功能(高亮显示,滚动,额外传参数)三个方面进行详细介绍。 一、功能特性 1.触发方式…...

Linux麒麟 V10 系统找回 root 密码的步骤
Linux麒麟 V10 系统找回 root 密码的步骤 1 环境介绍2 操作步骤2.1重启系统并进入 GRUB 菜单2.2 输入 GRUB 账户密码2.3 修改启动参数2.4 启动系统2.5 修改root 密码2.6 重启系统 3 Linux命令全方位指南实战教程Linux命令学习使用列表 1 环境介绍 有时候root 密码忘记…...
20、 DeepSeekMoE论文笔记
DeepSeekMoE 1、**研究背景与动机**2、传统MoE一、MoE架构核心原理详解1. **标准Transformer块的结构**2. **MoE层对FFN的替代**3. **稀疏性与计算效率** 二、举例说明:以 N 4 N4 N4 专家、 K 2 K2 K2 为例1. **场景设定**2. **亲和度计算与专家选择**3. **输出计…...
在 Spring Boot 中实现 WebSockets
什么是 WebSockets? WebSockets 是一种基于 TCP 的全双工通信协议,允许客户端和服务器之间建立持久的双向连接,用于实时数据交换。相较于传统的 HTTP 请求-响应模型,WebSockets 提供了低延迟、高效率的通信方式,特别适…...

stone 3d v3.3.0版本发布,含时间线和连接器等新功能
1.新加了时间线(timeline)编辑器,可以类似blender一样给对象制作动画 2.新加了度量(metrics)系统,通过scene对象检测器中的useMetrics属性来启用或禁用,启用时所选物体将显示三维度量数据 新加了…...

.whl文件
本文主要介绍了.whl文件的定义,怎么安装.whl文件(离线,在线)。 怎么查看cuda的版本,以及如何安装相应版本的cuda(本地电脑,超算上) 以及如何创建.whl文件 .whl文件的定义 Document…...

Git命令行中vim的操作
Git命令行用vim打开文件,或者用其他git命令打开了文件,需要编辑和保存文件等,有些命令表情奇怪,往往容易忘记这些命令。记录下。 下面这篇比较实用和简练: gitvim编辑文件命令 • Worktile社区https://worktile.com/…...

C#初级知识总结
一、什么是CIL 1.CIL(Common Intermidate Language)是指.Net的公共中间语言,它是一种编程语言。 .Net框架的各种语言在编译时都会编译成同一种中间语言(CIL),之后程序运行的时候CIL会被JIT(Just In Time)转换为二进制语言…...
使用 AI Agent 改善师生互动的设计文档
使用 AI Agent 改善师生互动的设计文档 一、引言 1.1 研究背景 当前教育领域的师生互动存在诸多挑战,如教师负担过重、学生个体差异大导致难以满足所有人的需求,以及信息传递延迟等问题。引入AI-Agent能够有效缓解这些问题,通过自动化手段协…...

Linux学习笔记之环境变量
写这篇博客的目的主要是因为本人学习动静态库时,用到了环境变量的知识,发现略有遗忘,因此回顾复习,整理成博客。 一、环境变量是什么 Linux环境变量是存储系统或程序运行时配置信息的特殊变量,用于为程序提供配置参数…...

16:00开始面试,16:08就出来了,问的问题有点变态。。。
从小厂出来,没想到在另一家公司又寄了。 到这家公司开始上班,加班是每天必不可少的,看在钱给的比较多的份上,就不太计较了。没想到4月一纸通知,所有人不准加班,加班费不仅没有了,薪资还要降40%…...
深度解析云计算:概念、优势与分类全览
以下是对云计算概念、优点和分类更详细的介绍: 一、云计算的概念 云计算是一种通过互联网提供计算服务的模式,它基于虚拟化、分布式计算、网络存储等一系列先进技术,将计算资源进行整合和管理,形成一个庞大的资源池。这些资源包…...

私钥连接服务器(已经有服务器私钥
前言:假设我们已经有了服务器的私钥,我们怎么配置呢? 下面我会从vsc的配置角度来写 ✅ 步骤一:准备工作 安装 VS Code(如果还没装) 👉 https://code.visualstudio.com/ 安装插件:Re…...

学员答题pk知识竞赛小程序怎么做
制作学员答题PK知识竞赛小程序,主要有以下步骤: 一、规划设计 明确需求:确定小程序的使用场景是校园知识竞赛、培训机构考核还是企业内部培训等。答题功能,规定答题的具体规则,包括题目类型(单选、多选、…...

外观模式:简化复杂系统接口的设计模式
外观模式:简化复杂系统接口的设计模式 一、模式核心:为复杂子系统提供统一简单接口 当一个系统由多个复杂子系统组成时(如电商系统中的支付、物流、库存模块),客户端直接调用子系统会导致依赖关系复杂、代码难以维护…...
vue3项目中eslint.config.ts配置rules
vue3项目中eslint.config.ts配置rules 1. 使用npm create vuelatest创建vue项目 默认的eslint.config.ts如下 import { globalIgnores } from eslint/config import { defineConfigWithVueTs, vueTsConfigs } from vue/eslint-config-typescript import pluginVue from esli…...

uniapp-商城-36-shop 购物车 选好了 进行订单确认2 支付方式颜色变化和颜色滤镜filter
颜色滤镜,在好多网页都这样使用,滤掉彩色,显示黑白,这在一些关键的日子中都这样使用。 1、依然回到订单确认页面 看到支付的颜色了嘛? <view class"payType"><view class"box" :class&q…...

Vue3 上传后的文件智能预览(实战体会)
目录 前言1. Demo12. Demo2 前言 🤟 找工作,来万码优才:👉 #小程序://万码优才/r6rqmzDaXpYkJZF 爬虫神器,无代码爬取,就来:bright.cn 此处的基本知识涉及较少,主要以Demo的形式供大…...
铃木一郎女儿是奥运会选手吗·棒球1号位
铃木一朗(Ichiro Suzuki) 铃木一朗职业生涯时间线 1973年出生于日本爱知县名古屋市。1992年以选秀第四顺位加入日本职棒(NPB)欧力士蓝浪队,开启职业棒球生涯。 1994-2000年 连续7年获得NPB太平洋联盟打击王ÿ…...
PyTorch与CUDA的关系
文章目录 前言一、如何查看PyTorch和torchvision的版本1.1 查看PyTorch版本1.2 查看torchvision版本二、如何确认PyTorch和torchvision是否支持CUDA加速2.1 检查PyTorch是否支持CUDA2.2 查看当前可用的GPU设备2.3 检查torchvision是否支持CUDA三、CUDA版本的秘密:为什么PyTorc…...