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

设计模式教程:备忘录模式(Memento Pattern)

备忘录模式(Memento Pattern)详解

一、模式概述

备忘录模式(Memento Pattern)是一种行为型设计模式,允许在不暴露对象实现细节的情况下,保存对象的内部状态,并在需要时恢复该状态。备忘录模式的核心思想是将对象的状态保存在一个独立的备忘录对象中,从而提供一种可以撤销操作、恢复历史状态的机制。

这种模式特别适用于需要提供“撤销/恢复”功能的应用场景,例如文本编辑器中的撤销操作、游戏中的进度保存和恢复等。

二、模式角色和类图

备忘录模式主要由三个角色组成:

  1. Originator(发起人): 负责创建一个备忘录,用以保存当前的内部状态,并可以恢复到之前的状态。
  2. Memento(备忘录): 存储发起人的内部状态,但不允许外部修改或查看这些状态。
  3. 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}
}

五、代码解释

  1. Memento 类: 存储状态并提供一个 getState() 方法来获取状态。它只负责数据的保存,不提供任何修改操作。

  2. Originator 类: 负责创建备忘录和恢复状态。它通过 createMemento() 方法创建一个新的备忘录对象,并通过 restoreMemento() 方法将状态恢复到备忘录中的状态。

  3. Caretaker 类: 负责保存和管理备忘录对象。它不直接修改备忘录的状态,只提供保存和获取备忘录的接口。

  4. 主函数: 在主函数中,我们创建了一个 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

八、优点与缺点

优点
  1. 封装性: 备忘录模式提供了很好的封装性,发起人的内部状态对外部不可见,外部只能通过备忘录来恢复状态,避免了直接暴露对象的实现细节。

  2. 支持撤销功能: 备忘录模式非常适用于实现撤销功能,能够轻松实现历史状态的恢复。

  3. 松耦合: CaretakerOriginator 之间的耦合较低,它们只通过备忘录进行通信,不直接互相依赖。

缺点
  1. 空间开销: 每次修改状态时都需要创建一个新的备忘录对象,这可能会带来额外的内存开销。尤其是当状态对象很大时,备忘录模式可能会成为性能瓶颈。

  2. 管理复杂性: 如果需要管理大量备忘录对象,可能会增加系统的复杂性。此时,需要提供有效的备忘录管理机制,避免内存泄漏或冗余备忘录的创建。

九、总结

备忘录模式通过引入备忘录对象来保存和恢复对象的状态,实现了一个灵活的历史记录和撤销恢复机制。它特别适用于需要保存对象状态并在后续恢复的场景,比如文本编辑器中的撤销操作、游戏中的进度保存等。尽管备忘录模式在实现时可能会带来额外的内存开销,但它为复杂系统中的状态管理提供了非常强大的支持。

希望这篇文章能够帮助你更好地理解备忘录模式,并在实际项目中合理地运用它。

版权声明
  1. 本文内容属于原创,欢迎转载,但请务必注明出处和作者,尊重原创版权。
  2. 转载时,请附带原文链接并注明“本文作者:扣丁梦想家
  3. 禁止未经授权的商业转载。

如果您有任何问题或建议,欢迎留言讨论。

相关文章:

设计模式教程:备忘录模式(Memento Pattern)

备忘录模式&#xff08;Memento Pattern&#xff09;详解 一、模式概述 备忘录模式&#xff08;Memento Pattern&#xff09;是一种行为型设计模式&#xff0c;允许在不暴露对象实现细节的情况下&#xff0c;保存对象的内部状态&#xff0c;并在需要时恢复该状态。备忘录模式…...

使用 C# 以api的形式调用 DeepSeek

一&#xff1a;创建 API 密钥 首先&#xff0c;您需要来自 DeepSeek 的 API 密钥。访问 DeepSeek&#xff0c;创建一个帐户&#xff0c;并生成一个新的 API 密钥。 二&#xff1a;安装所需的 NuGet 包 使用 NuGet 包管理器安装包&#xff0c;或在包管理器控制台中运行以下命…...

CS5366AN:高集成Type-C转HDMI 4K60Hz芯片的国产突破

一、芯片概述 CS5366AN 是集睿致远&#xff08;ASL&#xff09;推出的一款高度集成的 Type-C转HDMI 2.0视频转换芯片&#xff0c;专为扩展坞、游戏底座、高清显示设备等场景设计。其核心功能是将USB Type-C接口的DisplayPort信号&#xff08;DP Alt Mode&#xff09;转换为HDM…...

瑞芯微RK安卓Android主板GPIO按键配置方法,触觉智能嵌入式开发

触觉智能分享&#xff0c;瑞芯微RK安卓Android主板GPIO按键配置方法&#xff0c;方便大家更好利用空闲IO&#xff01;由触觉智能Purple Pi OH鸿蒙开发板演示&#xff0c;搭载了瑞芯微RK3566四核处理器&#xff0c;树莓派卡片电脑设计&#xff0c;支持安卓Android、开源鸿蒙Open…...

Dify自定义工作流集成指南:对接阿里云百炼文生图API的实现方案

dify工作流的应用基本解释 dify应用发布相关地址&#xff1a;应用发布 | Dify 根据官方教程&#xff0c;我们可以看到dify自定义的工作流可以发布为----工具 这个教程将介绍如何通过工作流建立一个使用阿里云百炼文生图模型。 工具则可以给其他功能使用&#xff0c;如agent…...

前端项目配置 Nginx 全攻略

在前端开发中&#xff0c;项目开发完成后&#xff0c;如何高效、稳定地将其部署到生产环境是至关重要的一步。Nginx 作为一款轻量级、高性能的 Web 服务器和反向代理服务器&#xff0c;凭借其出色的性能和丰富的功能&#xff0c;成为了前端项目部署的首选方案。本文将详细介绍在…...

基于开源鸿蒙(OpenHarmony)的【智能家居综合应用】系统

基于开源鸿蒙OpenHarmony的智能家居综合应用系统 1. 智能安防与门禁系统1) 系统概述2) 系统架构3&#xff09;关键功能实现4&#xff09;安全策略5&#xff09;总结 2.环境智能调节系统1&#xff09;场景描述2&#xff09;技术实现3&#xff09;总结 3.健康管理与睡眠监测1&…...

电子电气架构 --- 主机厂电子电气架构演进

我是穿拖鞋的汉子,魔都中坚持长期主义的汽车电子工程师。 老规矩,分享一段喜欢的文字,避免自己成为高知识低文化的工程师: 简单,单纯,喜欢独处,独来独往,不易合同频过着接地气的生活,除了生存温饱问题之外,没有什么过多的欲望,表面看起来很高冷,内心热情,如果你身…...

物联网通信应用案例之《智慧农业》

案例概述 在智慧农业方面&#xff0c;一般的应用场景为可以自动检测温度湿度等一系列环境情况并且可以自动做出相应的处理措施如简单的浇水和温度控制等&#xff0c;且数据情况可远程查看&#xff0c;以及用户可以实现远程控制。 基本实现原理 传感器通过串口将数据传递到Wi…...

Java注解的原理

目录 问题: 作用&#xff1a; 原理&#xff1a; 注解的限制 拓展&#xff1a; 问题: 今天刷面经&#xff0c;发现自己不懂注解的原理&#xff0c;特此记录。 作用&#xff1a; 注解的作用主要是给编译器看的&#xff0c;让它帮忙生成一些代码&#xff0c;或者是帮忙检查…...

AI知识架构之神经网络

神经网络:这是整个内容的主题,是一种模拟人类大脑神经元结构和功能的计算模型,在人工智能领域广泛应用。基本概念:介绍神经网络相关的基础概念,为后续深入理解神经网络做铺垫。定义与起源: 神经网络是模拟人类大脑神经元结构和功能的计算模型,其起源于对生物神经系统的研…...

OpenGL 04--GLSL、数据类型、Uniform、着色器类

一、着色器 在 OpenGL 中&#xff0c;着色器&#xff08;Shader&#xff09;是运行在 GPU 上的程序&#xff0c;用于处理图形渲染管线中的不同阶段。 这些小程序为图形渲染管线的某个特定部分而运行。从基本意义上来说&#xff0c;着色器只是一种把输入转化为输出的程序。着色器…...

学习笔记06——JVM调优

JVM 调优实战&#xff1a;性能优化的技巧与实战 在 Java 开发中&#xff0c;JVM&#xff08;Java Virtual Machine&#xff09;作为 Java 程序的运行环境&#xff0c;其性能直接影响到应用程序的响应速度和吞吐量。合理的 JVM 调优可以显著提升应用性能&#xff0c;降低延迟&a…...

深度学习(3)-TensorFlow入门(常数张量和变量)

低阶张量操作是所有现代机器学习的底层架构&#xff0c;可以转化为TensorFlow API。 张量&#xff0c;包括存储神经网络状态的特殊张量&#xff08;变量&#xff09;​。 张量运算&#xff0c;比如加法、relu、matmul。 反向传播&#xff0c;一种计算数学表达式梯度的方法&…...

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&#xff08;Composition API&#xff09; 2.选项式API&#xff08;Options…...

2025前端框架最新组件解析与实战技巧:Vue与React的革新之路

作者&#xff1a;飞天大河豚 引言 2025年的前端开发领域&#xff0c;Vue与React依然是开发者最青睐的框架。随着Vue 3的全面普及和React 18的持续优化&#xff0c;两大框架在组件化开发、性能优化、工程化支持等方面均有显著突破。本文将从最新组件特性、使用场景和编码技巧三…...

Elasticsearch 的分布式架构原理:通俗易懂版

Elasticsearch 的分布式架构原理&#xff1a;通俗易懂版 Lucene 和 Elasticsearch 的前世今生 Lucene 是一个功能强大的搜索库&#xff0c;提供了高效的全文检索能力。然而&#xff0c;直接基于 Lucene 开发非常复杂&#xff0c;即使是简单的功能也需要编写大量的 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 使用千帆…...

Java 语言特性(面试系列1)

一、面向对象编程 1. 封装&#xff08;Encapsulation&#xff09; 定义&#xff1a;将数据&#xff08;属性&#xff09;和操作数据的方法绑定在一起&#xff0c;通过访问控制符&#xff08;private、protected、public&#xff09;隐藏内部实现细节。示例&#xff1a; public …...

逻辑回归:给不确定性划界的分类大师

想象你是一名医生。面对患者的检查报告&#xff08;肿瘤大小、血液指标&#xff09;&#xff0c;你需要做出一个**决定性判断**&#xff1a;恶性还是良性&#xff1f;这种“非黑即白”的抉择&#xff0c;正是**逻辑回归&#xff08;Logistic Regression&#xff09;** 的战场&a…...

cf2117E

原题链接&#xff1a;https://codeforces.com/contest/2117/problem/E 题目背景&#xff1a; 给定两个数组a,b&#xff0c;可以执行多次以下操作&#xff1a;选择 i (1 < i < n - 1)&#xff0c;并设置 或&#xff0c;也可以在执行上述操作前执行一次删除任意 和 。求…...

【论文笔记】若干矿井粉尘检测算法概述

总的来说&#xff0c;传统机器学习、传统机器学习与深度学习的结合、LSTM等算法所需要的数据集来源于矿井传感器测量的粉尘浓度&#xff0c;通过建立回归模型来预测未来矿井的粉尘浓度。传统机器学习算法性能易受数据中极端值的影响。YOLO等计算机视觉算法所需要的数据集来源于…...

Mobile ALOHA全身模仿学习

一、题目 Mobile ALOHA&#xff1a;通过低成本全身远程操作学习双手移动操作 传统模仿学习&#xff08;Imitation Learning&#xff09;缺点&#xff1a;聚焦与桌面操作&#xff0c;缺乏通用任务所需的移动性和灵活性 本论文优点&#xff1a;&#xff08;1&#xff09;在ALOHA…...

Java线上CPU飙高问题排查全指南

一、引言 在Java应用的线上运行环境中&#xff0c;CPU飙高是一个常见且棘手的性能问题。当系统出现CPU飙高时&#xff0c;通常会导致应用响应缓慢&#xff0c;甚至服务不可用&#xff0c;严重影响用户体验和业务运行。因此&#xff0c;掌握一套科学有效的CPU飙高问题排查方法&…...

视觉slam十四讲实践部分记录——ch2、ch3

ch2 一、使用g++编译.cpp为可执行文件并运行(P30) g++ helloSLAM.cpp ./a.out运行 二、使用cmake编译 mkdir build cd build cmake .. makeCMakeCache.txt 文件仍然指向旧的目录。这表明在源代码目录中可能还存在旧的 CMakeCache.txt 文件,或者在构建过程中仍然引用了旧的路…...

基于 TAPD 进行项目管理

起因 自己写了个小工具&#xff0c;仓库用的Github。之前在用markdown进行需求管理&#xff0c;现在随着功能的增加&#xff0c;感觉有点难以管理了&#xff0c;所以用TAPD这个工具进行需求、Bug管理。 操作流程 注册 TAPD&#xff0c;需要提供一个企业名新建一个项目&#…...

快刀集(1): 一刀斩断视频片头广告

一刀流&#xff1a;用一个简单脚本&#xff0c;秒杀视频片头广告&#xff0c;还你清爽观影体验。 1. 引子 作为一个爱生活、爱学习、爱收藏高清资源的老码农&#xff0c;平时写代码之余看看电影、补补片&#xff0c;是再正常不过的事。 电影嘛&#xff0c;要沉浸&#xff0c;…...

【Android】Android 开发 ADB 常用指令

查看当前连接的设备 adb devices 连接设备 adb connect 设备IP 断开已连接的设备 adb disconnect 设备IP 安装应用 adb install 安装包的路径 卸载应用 adb uninstall 应用包名 查看已安装的应用包名 adb shell pm list packages 查看已安装的第三方应用包名 adb shell pm list…...