设计模式教程:备忘录模式(Memento Pattern)
备忘录模式(Memento Pattern)详解
一、模式概述
备忘录模式(Memento Pattern)是一种行为型设计模式,允许在不暴露对象实现细节的情况下,保存对象的内部状态,并在需要时恢复该状态。备忘录模式的核心思想是将对象的状态保存在一个独立的备忘录对象中,从而提供一种可以撤销操作、恢复历史状态的机制。
这种模式特别适用于需要提供“撤销/恢复”功能的应用场景,例如文本编辑器中的撤销操作、游戏中的进度保存和恢复等。
二、模式角色和类图
备忘录模式主要由三个角色组成:
- Originator(发起人): 负责创建一个备忘录,用以保存当前的内部状态,并可以恢复到之前的状态。
- Memento(备忘录): 存储发起人的内部状态,但不允许外部修改或查看这些状态。
- Caretaker(负责人): 负责保存和管理备忘录。Caretaker不能直接访问备忘录的内容,它只能保存和获取备忘录。
类图
+--------------------+
| Originator | <------------+
+--------------------+ |
| - state | |
| + createMemento() | |
| + restoreMemento() | |
+--------------------+ || |v |
+--------------------+ +-----------------+
| Memento | | Caretaker |
+--------------------+ +-----------------+
| - state | | - mementos |
| + getState() | | + addMemento() |
+--------------------+ | + getMemento() |+-----------------+
三、备忘录模式的关键概念
-
Originator: 负责保存和恢复对象的状态。它通过
createMemento()方法创建一个备忘录,并通过restoreMemento()方法恢复备忘录中的状态。 -
Memento: 储存一个对象的内部状态,且不允许外部直接修改。它仅提供获取状态的接口,而不会暴露对象内部的实现细节。
-
Caretaker: 负责保存和管理备忘录。它不直接操作或修改备忘录的内容,只能通过接口获取和存储备忘录。
四、代码实现
接下来通过一个简单的 Java 示例,详细说明备忘录模式的使用。
// Memento类,用于保存对象的状态
class Memento {private String state;public Memento(String state) {this.state = state;}public String getState() {return state;}
}// Originator类,负责创建和恢复备忘录
class Originator {private String state;public void setState(String state) {System.out.println("Setting state to: " + state);this.state = state;}public String getState() {return state;}// 创建一个备忘录,将当前状态保存public Memento createMemento() {return new Memento(state);}// 从备忘录恢复状态public void restoreMemento(Memento memento) {this.state = memento.getState();System.out.println("Restoring state to: " + state);}
}// Caretaker类,负责管理备忘录
class Caretaker {private Memento memento;// 保存备忘录public void saveMemento(Memento memento) {this.memento = memento;}// 获取备忘录public Memento getMemento() {return memento;}
}public class MementoPatternDemo {public static void main(String[] args) {// 创建发起人Originator originator = new Originator();// 创建负责人Caretaker caretaker = new Caretaker();// 设置状态并保存备忘录originator.setState("State #1");caretaker.saveMemento(originator.createMemento());// 设置另一个状态并保存备忘录originator.setState("State #2");caretaker.saveMemento(originator.createMemento());// 恢复状态originator.restoreMemento(caretaker.getMemento()); // 恢复到State #2originator.restoreMemento(caretaker.getMemento()); // 恢复到State #1}
}
五、代码解释
-
Memento 类: 存储状态并提供一个
getState()方法来获取状态。它只负责数据的保存,不提供任何修改操作。 -
Originator 类: 负责创建备忘录和恢复状态。它通过
createMemento()方法创建一个新的备忘录对象,并通过restoreMemento()方法将状态恢复到备忘录中的状态。 -
Caretaker 类: 负责保存和管理备忘录对象。它不直接修改备忘录的状态,只提供保存和获取备忘录的接口。
-
主函数: 在主函数中,我们创建了一个
Originator对象,设置其状态并创建了多个备忘录,然后通过Caretaker来管理备忘录,最后演示了如何通过备忘录恢复状态。
六、输出
Setting state to: State #1
Setting state to: State #2
Restoring state to: State #2
Restoring state to: State #1
七、实际应用场景
1. 文本编辑器的撤销功能
在文本编辑器中,每次用户修改文本时,程序都会保存当前的文本内容作为一个备忘录。当用户点击“撤销”按钮时,程序就可以恢复到用户修改前的状态。
// 伪代码示例
Editor editor = new Editor();
Caretaker caretaker = new Caretaker();editor.setContent("Hello World");
caretaker.saveMemento(editor.createMemento());editor.setContent("Hello Java");
caretaker.saveMemento(editor.createMemento());// 用户点击撤销按钮
editor.restoreMemento(caretaker.getMemento()); // 恢复到 "Hello World"
2. 游戏进度保存
在游戏中,玩家的进度可以定期保存为备忘录。玩家可以选择恢复到之前的某个保存点。
// 伪代码示例
GameProgress game = new GameProgress();
Caretaker caretaker = new Caretaker();game.setProgress("Level 1 - Score: 100");
caretaker.saveMemento(game.createMemento());game.setProgress("Level 2 - Score: 200");
caretaker.saveMemento(game.createMemento());// 玩家选择恢复
game.restoreMemento(caretaker.getMemento()); // 恢复到Level 1
八、优点与缺点
优点
-
封装性: 备忘录模式提供了很好的封装性,发起人的内部状态对外部不可见,外部只能通过备忘录来恢复状态,避免了直接暴露对象的实现细节。
-
支持撤销功能: 备忘录模式非常适用于实现撤销功能,能够轻松实现历史状态的恢复。
-
松耦合:
Caretaker和Originator之间的耦合较低,它们只通过备忘录进行通信,不直接互相依赖。
缺点
-
空间开销: 每次修改状态时都需要创建一个新的备忘录对象,这可能会带来额外的内存开销。尤其是当状态对象很大时,备忘录模式可能会成为性能瓶颈。
-
管理复杂性: 如果需要管理大量备忘录对象,可能会增加系统的复杂性。此时,需要提供有效的备忘录管理机制,避免内存泄漏或冗余备忘录的创建。
九、总结
备忘录模式通过引入备忘录对象来保存和恢复对象的状态,实现了一个灵活的历史记录和撤销恢复机制。它特别适用于需要保存对象状态并在后续恢复的场景,比如文本编辑器中的撤销操作、游戏中的进度保存等。尽管备忘录模式在实现时可能会带来额外的内存开销,但它为复杂系统中的状态管理提供了非常强大的支持。
希望这篇文章能够帮助你更好地理解备忘录模式,并在实际项目中合理地运用它。
版权声明
- 本文内容属于原创,欢迎转载,但请务必注明出处和作者,尊重原创版权。
- 转载时,请附带原文链接并注明“本文作者:扣丁梦想家
- 禁止未经授权的商业转载。
如果您有任何问题或建议,欢迎留言讨论。
相关文章:
设计模式教程:备忘录模式(Memento Pattern)
备忘录模式(Memento Pattern)详解 一、模式概述 备忘录模式(Memento Pattern)是一种行为型设计模式,允许在不暴露对象实现细节的情况下,保存对象的内部状态,并在需要时恢复该状态。备忘录模式…...
使用 C# 以api的形式调用 DeepSeek
一:创建 API 密钥 首先,您需要来自 DeepSeek 的 API 密钥。访问 DeepSeek,创建一个帐户,并生成一个新的 API 密钥。 二:安装所需的 NuGet 包 使用 NuGet 包管理器安装包,或在包管理器控制台中运行以下命…...
CS5366AN:高集成Type-C转HDMI 4K60Hz芯片的国产突破
一、芯片概述 CS5366AN 是集睿致远(ASL)推出的一款高度集成的 Type-C转HDMI 2.0视频转换芯片,专为扩展坞、游戏底座、高清显示设备等场景设计。其核心功能是将USB Type-C接口的DisplayPort信号(DP Alt Mode)转换为HDM…...
瑞芯微RK安卓Android主板GPIO按键配置方法,触觉智能嵌入式开发
触觉智能分享,瑞芯微RK安卓Android主板GPIO按键配置方法,方便大家更好利用空闲IO!由触觉智能Purple Pi OH鸿蒙开发板演示,搭载了瑞芯微RK3566四核处理器,树莓派卡片电脑设计,支持安卓Android、开源鸿蒙Open…...
Dify自定义工作流集成指南:对接阿里云百炼文生图API的实现方案
dify工作流的应用基本解释 dify应用发布相关地址:应用发布 | Dify 根据官方教程,我们可以看到dify自定义的工作流可以发布为----工具 这个教程将介绍如何通过工作流建立一个使用阿里云百炼文生图模型。 工具则可以给其他功能使用,如agent…...
前端项目配置 Nginx 全攻略
在前端开发中,项目开发完成后,如何高效、稳定地将其部署到生产环境是至关重要的一步。Nginx 作为一款轻量级、高性能的 Web 服务器和反向代理服务器,凭借其出色的性能和丰富的功能,成为了前端项目部署的首选方案。本文将详细介绍在…...
基于开源鸿蒙(OpenHarmony)的【智能家居综合应用】系统
基于开源鸿蒙OpenHarmony的智能家居综合应用系统 1. 智能安防与门禁系统1) 系统概述2) 系统架构3)关键功能实现4)安全策略5)总结 2.环境智能调节系统1)场景描述2)技术实现3)总结 3.健康管理与睡眠监测1&…...
电子电气架构 --- 主机厂电子电气架构演进
我是穿拖鞋的汉子,魔都中坚持长期主义的汽车电子工程师。 老规矩,分享一段喜欢的文字,避免自己成为高知识低文化的工程师: 简单,单纯,喜欢独处,独来独往,不易合同频过着接地气的生活,除了生存温饱问题之外,没有什么过多的欲望,表面看起来很高冷,内心热情,如果你身…...
物联网通信应用案例之《智慧农业》
案例概述 在智慧农业方面,一般的应用场景为可以自动检测温度湿度等一系列环境情况并且可以自动做出相应的处理措施如简单的浇水和温度控制等,且数据情况可远程查看,以及用户可以实现远程控制。 基本实现原理 传感器通过串口将数据传递到Wi…...
Java注解的原理
目录 问题: 作用: 原理: 注解的限制 拓展: 问题: 今天刷面经,发现自己不懂注解的原理,特此记录。 作用: 注解的作用主要是给编译器看的,让它帮忙生成一些代码,或者是帮忙检查…...
AI知识架构之神经网络
神经网络:这是整个内容的主题,是一种模拟人类大脑神经元结构和功能的计算模型,在人工智能领域广泛应用。基本概念:介绍神经网络相关的基础概念,为后续深入理解神经网络做铺垫。定义与起源: 神经网络是模拟人类大脑神经元结构和功能的计算模型,其起源于对生物神经系统的研…...
OpenGL 04--GLSL、数据类型、Uniform、着色器类
一、着色器 在 OpenGL 中,着色器(Shader)是运行在 GPU 上的程序,用于处理图形渲染管线中的不同阶段。 这些小程序为图形渲染管线的某个特定部分而运行。从基本意义上来说,着色器只是一种把输入转化为输出的程序。着色器…...
学习笔记06——JVM调优
JVM 调优实战:性能优化的技巧与实战 在 Java 开发中,JVM(Java Virtual Machine)作为 Java 程序的运行环境,其性能直接影响到应用程序的响应速度和吞吐量。合理的 JVM 调优可以显著提升应用性能,降低延迟&a…...
深度学习(3)-TensorFlow入门(常数张量和变量)
低阶张量操作是所有现代机器学习的底层架构,可以转化为TensorFlow API。 张量,包括存储神经网络状态的特殊张量(变量)。 张量运算,比如加法、relu、matmul。 反向传播,一种计算数学表达式梯度的方法&…...
3-2 WPS JS宏 工作簿的打开与保存(模板批量另存为工作)学习笔记
************************************************************************************************************** 点击进入 -我要自学网-国内领先的专业视频教程学习网站 *******************************************************************************************…...
【GO】学习笔记
目录 学习链接 开发环境 开发工具 GVM - GO多版本部署 GOPATH 与 go.mod go常用命令 环境初始化 编译与运行 GDB -- GNU 调试器 基本语法与字符类型 关键字与标识符 格式化占位符 基本语法 初始值&零值&默认值 变量声明与赋值 _ 下划线的用法 字…...
【TypeScript】ts在vue中的使用
目录 一、Vue 3 TypeScript 1. 项目创建与配置 项目创建 关键配置文件 2.完整项目结构示例 3. 组件 Props 类型定义 4. 响应式数据与 Ref 5. Composition 函数复用 二、组件开发 1.组合式API(Composition API) 2.选项式API(Options…...
2025前端框架最新组件解析与实战技巧:Vue与React的革新之路
作者:飞天大河豚 引言 2025年的前端开发领域,Vue与React依然是开发者最青睐的框架。随着Vue 3的全面普及和React 18的持续优化,两大框架在组件化开发、性能优化、工程化支持等方面均有显著突破。本文将从最新组件特性、使用场景和编码技巧三…...
Elasticsearch 的分布式架构原理:通俗易懂版
Elasticsearch 的分布式架构原理:通俗易懂版 Lucene 和 Elasticsearch 的前世今生 Lucene 是一个功能强大的搜索库,提供了高效的全文检索能力。然而,直接基于 Lucene 开发非常复杂,即使是简单的功能也需要编写大量的 Java 代码&…...
【DeepSeek】【GPT-Academic】:DeepSeek集成到GPT-Academic(官方+第三方)
目录 1 官方deepseek 1.1 拉取学术GPT项目 1.2 安装依赖 1.3 修改配置文件中的DEEPSEEK_API_KEY 2 第三方API 2.1 修改DEEPSEEK_API_KEY 2.2 修改CUSTOM_API_KEY_PATTERM 2.3 地址重定向 2.4 修改模型参数 2.5 成功调用 2.6 尝试添加一个deepseek-r1参数 3 使用千帆…...
跨链模式:多链互操作架构与性能扩展方案
跨链模式:多链互操作架构与性能扩展方案 ——构建下一代区块链互联网的技术基石 一、跨链架构的核心范式演进 1. 分层协议栈:模块化解耦设计 现代跨链系统采用分层协议栈实现灵活扩展(H2Cross架构): 适配层…...
GitHub 趋势日报 (2025年06月08日)
📊 由 TrendForge 系统生成 | 🌐 https://trendforge.devlive.org/ 🌐 本日报中的项目描述已自动翻译为中文 📈 今日获星趋势图 今日获星趋势图 884 cognee 566 dify 414 HumanSystemOptimization 414 omni-tools 321 note-gen …...
三体问题详解
从物理学角度,三体问题之所以不稳定,是因为三个天体在万有引力作用下相互作用,形成一个非线性耦合系统。我们可以从牛顿经典力学出发,列出具体的运动方程,并说明为何这个系统本质上是混沌的,无法得到一般解…...
dify打造数据可视化图表
一、概述 在日常工作和学习中,我们经常需要和数据打交道。无论是分析报告、项目展示,还是简单的数据洞察,一个清晰直观的图表,往往能胜过千言万语。 一款能让数据可视化变得超级简单的 MCP Server,由蚂蚁集团 AntV 团队…...
Mobile ALOHA全身模仿学习
一、题目 Mobile ALOHA:通过低成本全身远程操作学习双手移动操作 传统模仿学习(Imitation Learning)缺点:聚焦与桌面操作,缺乏通用任务所需的移动性和灵活性 本论文优点:(1)在ALOHA…...
【VLNs篇】07:NavRL—在动态环境中学习安全飞行
项目内容论文标题NavRL: 在动态环境中学习安全飞行 (NavRL: Learning Safe Flight in Dynamic Environments)核心问题解决无人机在包含静态和动态障碍物的复杂环境中进行安全、高效自主导航的挑战,克服传统方法和现有强化学习方法的局限性。核心算法基于近端策略优化…...
CSS | transition 和 transform的用处和区别
省流总结: transform用于变换/变形,transition是动画控制器 transform 用来对元素进行变形,常见的操作如下,它是立即生效的样式变形属性。 旋转 rotate(角度deg)、平移 translateX(像素px)、缩放 scale(倍数)、倾斜 skewX(角度…...
OD 算法题 B卷【正整数到Excel编号之间的转换】
文章目录 正整数到Excel编号之间的转换 正整数到Excel编号之间的转换 excel的列编号是这样的:a b c … z aa ab ac… az ba bb bc…yz za zb zc …zz aaa aab aac…; 分别代表以下的编号1 2 3 … 26 27 28 29… 52 53 54 55… 676 677 678 679 … 702 703 704 705;…...
关于easyexcel动态下拉选问题处理
前些日子突然碰到一个问题,说是客户的导入文件模版想支持部分导入内容的下拉选,于是我就找了easyexcel官网寻找解决方案,并没有找到合适的方案,没办法只能自己动手并分享出来,针对Java生成Excel下拉菜单时因选项过多导…...
uniapp 集成腾讯云 IM 富媒体消息(地理位置/文件)
UniApp 集成腾讯云 IM 富媒体消息全攻略(地理位置/文件) 一、功能实现原理 腾讯云 IM 通过 消息扩展机制 支持富媒体类型,核心实现方式: 标准消息类型:直接使用 SDK 内置类型(文件、图片等)自…...
