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

Java设计模式之备忘录模式详解

Java设计模式之备忘录模式详解


一、备忘录模式核心思想

核心目标捕获对象内部状态并在需要时恢复,同时不破坏对象的封装性。如同游戏存档系统,允许玩家保存当前进度并在需要时回退到之前的状态。


二、备忘录模式类图(Mermaid)

创建
存储
Originator
-state: String
+saveToMemento() : Memento
+restoreFromMemento(Memento)
+setState(String)
+getState() : String
Memento
-state: String
+getState() : String
+setState(String)
Caretaker
-mementos: List
+addMemento(Memento)
+getMemento(int) : Memento

三、代码实现示例

1. 文本编辑器撤销功能

// 备忘录类(存储编辑器状态)
class TextMemento {private final String text;private final int cursorPosition;public TextMemento(String text, int cursorPosition) {this.text = text;this.cursorPosition = cursorPosition;}public String getText() {return text;}public int getCursorPosition() {return cursorPosition;}
}// 原发器:文本编辑器
class TextEditor {private StringBuilder text = new StringBuilder();private int cursorPosition = 0;public void type(String words) {text.insert(cursorPosition, words);cursorPosition += words.length();System.out.println("当前文本: " + text);}public void moveCursor(int position) {cursorPosition = Math.max(0, Math.min(position, text.length()));System.out.println("光标移动到: " + cursorPosition);}public TextMemento save() {return new TextMemento(text.toString(), cursorPosition);}public void restore(TextMemento memento) {this.text = new StringBuilder(memento.getText());this.cursorPosition = memento.getCursorPosition();System.out.println("恢复文本: " + text);}public void printStatus() {System.out.println("文本: " + text);System.out.println("光标位置: " + cursorPosition);}
}// 管理者:历史记录
class History {private List<TextMemento> states = new ArrayList<>();public void push(TextMemento state) {states.add(state);}public TextMemento pop() {if (states.isEmpty()) return null;return states.remove(states.size() - 1);}public TextMemento get(int index) {return states.get(index);}
}// 客户端调用
public class Client {public static void main(String[] args) {TextEditor editor = new TextEditor();History history = new History();// 编辑文本editor.type("设计模式");history.push(editor.save());  // 保存状态1editor.type("备忘录");history.push(editor.save());  // 保存状态2editor.type("示例");System.out.println("\n当前状态:");editor.printStatus();// 撤销到上一步System.out.println("\n撤销操作:");editor.restore(history.pop());  // 恢复状态2editor.printStatus();// 再撤销一步System.out.println("\n再次撤销:");editor.restore(history.pop());  // 恢复状态1editor.printStatus();}
}

四、模式优缺点分析

✅ 优势

  • 状态封装:不暴露对象内部实现细节
  • 撤销/重做支持:轻松实现历史记录功能
  • 状态快照:支持任意时刻状态保存
  • 符合单一职责:状态管理职责分离

❌ 缺点

  • 内存消耗:大量状态保存可能导致内存占用高
  • 性能影响:大对象状态保存/恢复可能耗时
  • 复杂状态处理:嵌套对象状态保存较复杂

五、典型应用场景

  1. 文本编辑器:撤销/重做功能实现
  2. 游戏开发:保存/加载游戏进度
  3. 事务回滚:数据库操作回退
  4. 软件配置:保存和恢复用户设置
  5. 绘图软件:操作历史记录
  6. 状态机:回退到之前状态

六、Mermaid序列图(状态保存与恢复)

Client Originator Caretaker Memento 修改状态 saveToMemento() 创建备忘录(状态) 返回备忘录 addMemento(备忘录) getMemento() 备忘录 restoreFromMemento(备忘录) getState() 状态值 Client Originator Caretaker Memento

七、备忘录模式 vs 其他模式

对比模式核心区别
命令模式封装操作请求,可支持撤销
状态模式对象行为随状态改变
原型模式克隆对象而非保存状态

八、实际框架应用案例

1. Java Swing的UndoManager

管理
«interface»
UndoableEdit
+undo()
+redo()
AbstractUndoableEdit
UndoManager
CompoundEdit

2. Spring框架的事务管理

@Transactional
public void transferMoney(Account from, Account to, double amount) {// 事务开始时创建备忘录(保存点)savepoint = TransactionAspectSupport.currentTransactionStatus().createSavepoint();try {from.withdraw(amount);to.deposit(amount);} catch (Exception e) {// 回滚到保存点TransactionAspectSupport.currentTransactionStatus().rollbackToSavepoint(savepoint);}
}

九、高级应用技巧

1. 增量备忘录(节省内存)

class IncrementalMemento {private final String diff;  // 只存储变化部分public IncrementalMemento(String diff) {this.diff = diff;}public String apply(String base) {// 应用差异到基础状态return base + diff;}
}

2. 多级撤销/重做栈

class HistoryManager {private Stack<Memento> undoStack = new Stack<>();private Stack<Memento> redoStack = new Stack<>();public void save(Memento state) {undoStack.push(state);redoStack.clear();}public Memento undo() {if (!undoStack.isEmpty()) {Memento state = undoStack.pop();redoStack.push(state);return undoStack.isEmpty() ? null : undoStack.peek();}return null;}public Memento redo() {if (!redoStack.isEmpty()) {Memento state = redoStack.pop();undoStack.push(state);return state;}return null;}
}

十、常见问题解答

Q1:如何保存复杂对象状态?

  • 序列化:实现Serializable接口
class ComplexMemento implements Serializable {private Object complexState;
}

Q2:如何处理外部资源引用?

使用深拷贝避免外部资源影响:

class ResourceMemento {private Resource resourceCopy;public ResourceMemento(Resource original) {this.resourceCopy = original.deepCopy();}
}

Q3:如何限制备忘录访问权限?

使用内部类实现封装:

class Originator {private String state;// 内部备忘录类class Memento {private String state;private Memento(String state) {this.state = state;}private String getState() {return state;}}public Memento save() {return new Memento(state);}
}

相关文章:

Java设计模式之备忘录模式详解

Java设计模式之备忘录模式详解 一、备忘录模式核心思想 核心目标&#xff1a;捕获对象内部状态并在需要时恢复&#xff0c;同时不破坏对象的封装性。如同游戏存档系统&#xff0c;允许玩家保存当前进度并在需要时回退到之前的状态。 二、备忘录模式类图&#xff08;Mermaid&am…...

Azure DevOps Server 2022.2 补丁(Patch 5)

微软Azure DevOps Server的产品组在4月8日发布了2022.2 的第5个补丁。下载路径为&#xff1a;https://aka.ms/devops2022.2patch5 这个补丁的主要功能是修改了代理(Agent)二进制安装文件的下载路径&#xff1b;之前&#xff0c;微软使用这个CND(域名为vstsagentpackage.azuree…...

手摸手还原vue3中reactive的get陷阱以及receiver的作用

提示&#xff1a;文章写完后&#xff0c;目录可以自动生成&#xff0c;如何生成可参考右边的帮助文档 文章目录 前言一、实例是什么&#xff1f;二、new Prxoy三、实现代码1.引入代码2.读入数据 总结 前言 receiver不是为解决get陷阱而生&#xff0c;而是为解决Proxy中的this绑…...

小明的Java面试奇遇之互联网保险系统架构与性能优化

一、文章标题 小明的Java面试奇遇之互联网保险系统架构与性能优化&#x1f680; 二、文章标签 Java,Spring Boot,MyBatis,Redis,Kafka,JVM,多线程,互联网保险,系统架构,性能优化 三、文章概述 本文模拟了程序员小明在应聘互联网保险系统开发岗位时&#xff0c;参与的一场深…...

C++学习-入门到精通【13】标准库的容器和迭代器

C学习-入门到精通【13】标准库的容器和迭代器 目录 C学习-入门到精通【13】标准库的容器和迭代器一、标准模板库简介1.容器简介2.STL容器总览3.近容器4.STL容器的通用函数5.首类容器的通用typedef6.对容器元素的要求 二、迭代器简介1.使用istream_iterator输入&#xff0c;使用…...

C# 面向对象特性

面向对象编程的三大基本特性是&#xff1a;封装、继承和多态。下面将详细介绍这三大特性在C#中的体现方式。 封装 定义&#xff1a;把对象的数据和操作代码组合在同一个结构中&#xff0c;这就是对象的封装性。 体现方式&#xff1a; 使用访问修饰符控制成员的可见性 通过属…...

ElasticStack技术之logstash介绍

一、什么是Logstash Logstash 是 Elastic Stack&#xff08;ELK Stack&#xff09;中的一个开源数据处理管道工具&#xff0c;主要用于收集、解析、过滤和传输数据。它支持多种输入源&#xff0c;如文件、网络、数据库等&#xff0c;能够灵活地对数据进行处理&#xff0c;比如…...

前端与后端

实例一 处理登录页面请求 # 处理登录页面请求 app.route(/c, methods[GET, POST]) # /c是网页地址 def login(): usernameaa passwordbb print(username,password) if request.method POST: username request.form.get(yhm) password requ…...

CI/CD 持续集成、持续交付、持续部署

CI/CD 是 持续集成&#xff08;Continuous Integration&#xff09; 和 持续交付/持续部署&#xff08;Continuous Delivery/Deployment&#xff09; 的缩写&#xff0c;代表现代软件开发中通过自动化流程快速、可靠地构建、测试和发布代码的实践。其核心目标是 减少人工干预、…...

代码随想录60期day54

岛屿dfs #include<iostream> #include<vector> using namespace std;int dir[4][2] {0,1,1,0,-1,0,0,-1};void dfs(const vector<vector<int>>&grid,vector<vecotr<bool>>&visited,int x,int y){for(int i 0 ; i < 4; i){in…...

关于easyx头文件

一、窗口创建 &#xff08;1&#xff09;几种创建方式 #include<easyx.h>//easyx的头文件 #include<iostream> using namespace std;int main() {//创建一个500*500的窗口//参数为&#xff1a;长度&#xff0c;宽度&#xff0c;是否显示黑框&#xff08;无参为不…...

Java 中执行命令并使用指定配置文件的最佳实践

在Java开发中&#xff0c;有时需要从Java应用程序中执行系统命令&#xff0c;并使用指定的配置文件来控制这些命令的行为。本文将详细介绍在Java中执行命令并使用指定配置文件的最佳实践&#xff0c;包括如何设置环境变量、重定向输入输出以及处理可能出现的异常。 一、基本实…...

django入门-orm数据库操作

一&#xff1a;下载数据库依赖项mysqlclient pip install mysqlclient 二&#xff1a;django配置文件配置数据库链接 路径&#xff1a;mysite2\mysite2\settings.py DATABASES {default: {ENGINE: django.db.backends.mysql,NAME: data, # 数据库名称USER: root, …...

​​食品电商突围战!品融电商全平台代运营,助您抢占天猫京东抖音红利!

​​食品电商突围战&#xff01;品融电商全平台代运营&#xff0c;助您抢占天猫京东抖音红利&#xff01;​​ ​​一、食品电商的黄金时代&#xff1a;机遇与挑战并存​​ 随着消费升级和线上渗透率的持续攀升&#xff0c;食品行业正迎来前所未有的发展机遇。2023年&#xff…...

Termux下如何使用MATLAB

实际上&#xff0c;termux 目前无法运行MATLAB&#xff0c;但是可以运行MATLAB的平替octave &#xff0c;可以完全在终端环境运行&#xff0c;方便运算和查看模型拟合结果等&#xff0c;完全兼容MATLAB命令。 食用方法&#xff1a; //pkg install wget wget https://its-poin…...

STM32外部中断(EXTI)以及旋转编码器的简介

一、外部中断机制概述 中断是指当主程序执行期间出现特定触发条件&#xff08;即中断源&#xff09;时&#xff0c;CPU将暂停当前任务&#xff0c;转而执行相应的中断服务程序&#xff08;ISR&#xff09;&#xff0c;待处理完成后恢复原程序的运行流程。该机制通过事件驱动…...

双擎驱动:华为云数字人与DeepSeek大模型的智能交互升级方案

一、技术融合概述 华为云数字人 华为云数字人&#xff0c;全称&#xff1a;数字内容生产线 MetaStudio。数字内容生产线&#xff0c;提供数字人视频制作、视频直播、智能交互、企业代言等多种服务能力&#xff0c;使能千行百业降本增效。另外&#xff0c;数字内容生产线&#…...

Unity Version Control UVC报错:Not connected. Trying to re-connect…

问题背景 今天备份项目的时候遇到了这个问题&#xff0c;起因是Unity停用了原始的Plastic SCM的项目管理功能&#xff0c;我使用新的Unity Version Control系统时遇到了无法新建workspace的问题&#xff0c;即使新建之后进入Unity也无法连接到仓库&#xff0c;点击重试也无反应…...

场景题-1

场景题-1 订单到期关闭 1、DelayQueue 无界阻塞队列&#xff0c;用于放置实现了Delayed接口的对象&#xff0c;基于PriorityQueue实现&#xff0c;可用于实现在指定的延迟时间之后处理元素。订单创建后放入队列中&#xff0c;然后使用一个常驻任务不停地执行扫描取出超时订单…...

Java复习Day26

Lambda表达式简介 Lambda表达式是Java 8的重要特性&#xff0c;允许使用简洁的表达式代替功能接口。它类似于方法&#xff0c;包含参数列表和执行主体&#xff08;可以是表达式或代码块&#xff09;。Lambda可以视为匿名内部类的语法糖&#xff0c;也被称为闭包。 优点 代码…...

实验设计与分析(第6版,Montgomery)第5章析因设计引导5.7节思考题5.5 R语言解题

本文是实验设计与分析&#xff08;第6版&#xff0c;Montgomery著&#xff0c;傅珏生译) 第5章析因设计引导5.7节思考题5.5 R语言解题。主要涉及方差分析&#xff0c;正态假设检验&#xff0c;残差分析&#xff0c;交互作用图。 dataframe <-data.frame( wrapc(17,20,12,9,…...

阿里云百炼全解析:一站式大模型开发平台的架构与行业实践

目录 大模型开发范式的革新平台核心架构与技术解析全生命周期开发工作流企业级安全与合规体系行业应用场景与最佳实践未来演进与技术展望1. 大模型开发范式的革新 1.1 从碎片化到平台化的演进 传统大模型开发面临三大核心挑战:算力管理复杂、工具链割裂、安全合规风险高。阿…...

字节新出的MCP应用DeepSearch,有点意思。

大家好&#xff0c;我是苍何。 悄悄告诉你个事&#xff0c;昨天我去杭州参加字节火山方舟举办的开发者见面会了&#xff0c;你别说&#xff0c;还真有点刘姥姥进大观园的感觉&#x1f436; 现场真实体验完这次新发布的产品和模型&#xff0c;激动的忍不住想给大家做一波分享。…...

​​Agentic Voice Stack 热门项目

以下是当前在 ​​Agentic Voice Stack 工作流​​领域较为热门的开源项目&#xff0c;涵盖语音交互、多模态控制、工作流编排等核心能力&#xff0c;综合多个权威来源整理而成&#xff1a; &#x1f399;️ ​​一、语音交互层&#xff08;Speech-to-Speech & Text-to-Spe…...

机器学习在多介质环境中多污染物空间预测的应用研究

机器学习在多介质环境中多污染物空间预测的应用研究 1. 引言 1.1 研究背景与意义 随着工业化和城市化进程加速,环境中多种污染物的共存已成为全球性环境问题。重金属(如铅、汞、镉)、有机污染物(如多环芳烃、农药残留)和新兴污染物(如微塑料、药品残留)在空气、水体、…...

期货反向跟单运营逻辑推导思路

期货反向跟单运营逻辑推导思路 很多刚接触期货反向跟的朋友第一印象就是&#xff1a;这绝对是一个完美的策略&#xff0c;在认知不到位的情况下就开始运营&#xff0c;结果就是赔的稀里哗啦。然后告诉身边所有的人&#xff0c;期货反向跟单不靠谱。 这就是一个很有意思的事情&…...

使用 HTML + JavaScript 实现图片裁剪上传功能

本文将详细介绍一个基于 HTML 和 JavaScript 实现的图片裁剪上传功能。该功能支持文件选择、拖放上传、图片预览、区域选择、裁剪操作以及图片下载等功能&#xff0c;适用于需要进行图片处理的 Web 应用场景。 效果演示 项目概述 本项目主要包含以下核心功能&#xff1a; 文…...

Redis 缓存粒度如何控制?缓存整个对象还是部分字段?

控制 Redis 缓存粒度&#xff0c;即决定是缓存整个对象还是对象的部分字段&#xff0c;是一个需要在性能、内存使用、数据一致性、更新复杂性和开发成本之间进行权衡的决策。没有绝对的“最佳”方案&#xff0c;需要根据具体业务场景来选择。 以下是两种主要策略及其优缺点&am…...

【灵动Mini-F5265-OB】vscode+gcc工程创建、下载、调试

【前言】 【灵动Mini-F5265-OB】在官方的例程中提供了mdk、IAR的开发环境&#xff0c;使用起来非常方便。有位大佬也提供了一个gcc的示例&#xff0c;但是我使用vscode的keil插件进行工程创建&#xff0c;但是提示pack是对不上的。所以我决定重新创建我的vscode来创建开发环境。…...

程序设计实践期末考试模拟题(1)

1、排列论文 #include<bits/stdc.h> using namespace std; const int N105; vector<int>g[N]; int a[N]; int n,m; int flag; int topSort(){queue<int>q;for(int i1;i<n;i){if(a[i]0){q.push(i);}}int cnt0;flag1;while(!q.empty()){int tq.front();q.p…...