备忘录模式:实现对象状态撤销与恢复的设计模式

news2025/4/27 3:18:36

备忘录模式:实现对象状态撤销与恢复的设计模式

一、模式核心:在不破坏封装性的前提下保存和恢复对象状态

在软件开发中,经常需要实现 “撤销” 功能(如文本编辑器的撤销修改、游戏存档读取)。直接暴露对象内部状态会破坏封装性,而备忘录模式通过独立的备忘录对象封装状态,实现安全的状态管理。

备忘录模式(Memento Pattern) 允许在不暴露对象内部细节的情况下,捕获对象的内部状态并保存为备忘录(Memento),后续可通过备忘录恢复对象状态。核心解决:

  • 状态封装:备忘录对象封装对象状态,避免外部直接访问内部属性。
  • 撤销 / 重做支持:通过保存多个备忘录实现多步撤销(如版本控制)。
  • 单一职责分离:将状态管理逻辑从原发对象中分离,符合开闭原则。

核心思想与 UML 类图(PlantUML 语法)

备忘录模式包含以下角色:

  1. 原发器(Originator):创建并恢复自身状态的对象。
  2. 备忘录(Memento):存储原发器的状态,提供有限访问接口。
  3. 管理者(Caretaker):管理备忘录的创建、存储和获取(如历史记录列表)。

PlantUML Diagram

二、核心实现:文本编辑器的撤销功能

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 命令模式

类型核心功能协作方式
备忘录模式保存 / 恢复对象状态原发器创建备忘录,管理者存储
命令模式封装操作命令,支持撤销 / 重做命令对象记录操作前后状态
组合使用命令模式调用备忘录模式保存操作状态,实现多步撤销命令执行时创建备忘录,撤销时恢复

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

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

相关文章

freecad参数化三维模型装配体解析至web端,切换参数组或修改参数

用免费开源的freecad制作全参数化的三维模型&#xff0c;并且装配&#xff0c;上传至服务器&#xff0c;解析至web端&#xff0c;用户可以切换参数或修改参数&#xff0c;驱动模型改变。 freecad全参数化装配体模型解析至web端进行参数切换、修改完整展示_哔哩哔哩_bilibili …

浅析锁的应用与场景

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

语音合成之五语音合成中的“一对多”问题主流模型解决方案分析

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

ElementUi的Dropdown下拉菜单的详细介绍及使用

Dropdown是 ElementUI 中用于创建下拉菜单项的一个组件&#xff0c;通常el-dropdown-item 包裹在 el-dropdown 组件中使用。以下从功能特性(一些属性及方法)、使用和高级功能(高亮显示&#xff0c;滚动&#xff0c;额外传参数)三个方面进行详细介绍。 一、功能特性 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 密码忘记&#xf…

stone 3d v3.3.0版本发布,含时间线和连接器等新功能

1.新加了时间线&#xff08;timeline&#xff09;编辑器&#xff0c;可以类似blender一样给对象制作动画 2.新加了度量&#xff08;metrics&#xff09;系统&#xff0c;通过scene对象检测器中的useMetrics属性来启用或禁用&#xff0c;启用时所选物体将显示三维度量数据 新加了…

.whl文件

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

Git命令行中vim的操作

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

C#初级知识总结

一、什么是CIL 1.CIL(Common Intermidate Language)是指.Net的公共中间语言&#xff0c;它是一种编程语言。 .Net框架的各种语言在编译时都会编译成同一种中间语言&#xff08;CIL&#xff09;&#xff0c;之后程序运行的时候CIL会被JIT(Just In Time)转换为二进制语言&#xf…

Linux学习笔记之环境变量

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

16:00开始面试,16:08就出来了,问的问题有点变态。。。

从小厂出来&#xff0c;没想到在另一家公司又寄了。 到这家公司开始上班&#xff0c;加班是每天必不可少的&#xff0c;看在钱给的比较多的份上&#xff0c;就不太计较了。没想到4月一纸通知&#xff0c;所有人不准加班&#xff0c;加班费不仅没有了&#xff0c;薪资还要降40%…

私钥连接服务器(已经有服务器私钥

前言&#xff1a;假设我们已经有了服务器的私钥&#xff0c;我们怎么配置呢&#xff1f; 下面我会从vsc的配置角度来写 ✅ 步骤一&#xff1a;准备工作 安装 VS Code&#xff08;如果还没装&#xff09; &#x1f449; https://code.visualstudio.com/ 安装插件&#xff1a;Re…

学员答题pk知识竞赛小程序怎么做

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

外观模式:简化复杂系统接口的设计模式

外观模式&#xff1a;简化复杂系统接口的设计模式 一、模式核心&#xff1a;为复杂子系统提供统一简单接口 当一个系统由多个复杂子系统组成时&#xff08;如电商系统中的支付、物流、库存模块&#xff09;&#xff0c;客户端直接调用子系统会导致依赖关系复杂、代码难以维护…

uniapp-商城-36-shop 购物车 选好了 进行订单确认2 支付方式颜色变化和颜色滤镜filter

颜色滤镜&#xff0c;在好多网页都这样使用&#xff0c;滤掉彩色&#xff0c;显示黑白&#xff0c;这在一些关键的日子中都这样使用。 1、依然回到订单确认页面 看到支付的颜色了嘛&#xff1f; <view class"payType"><view class"box" :class&q…

Vue3 上传后的文件智能预览(实战体会)

目录 前言1. Demo12. Demo2 前言 &#x1f91f; 找工作&#xff0c;来万码优才&#xff1a;&#x1f449; #小程序://万码优才/r6rqmzDaXpYkJZF 爬虫神器&#xff0c;无代码爬取&#xff0c;就来&#xff1a;bright.cn 此处的基本知识涉及较少&#xff0c;主要以Demo的形式供大…

CCE13.【C++ Cont】练习题组13 静态链表专题

目录 1.B3630 排队顺序 题目 分析 代码 提交结果 2.B3631 单向链表 题目 分析 前置知识:map数组加快访问速度(简单的哈希表优化) 使用map数组的重要提醒 代码 提交结果 3.★P1160 队列安排 题目 分析 方法1:带头不循环双向链表的设计 方法2:带头循环的双向链表…

内联函数(c++)

预处理&#xff1a;优点&#xff1a;内嵌到目标代码&#xff0c;减少函数的调用。 缺点&#xff1a;在预处理阶段完成替换&#xff0c;避免了语义上的差错。 egg&#xff1a; #define SQR(X) ((X)*(X)) 函数&#xff1a;优点&#xff1a;完成了某一类操作的抽象&#xff0c;…

R7周:糖尿病预测模型优化探索

&#x1f368; 本文为&#x1f517;365天深度学习训练营中的学习记录博客 &#x1f356; 原作者&#xff1a;K同学啊 一、数据预处理 1.设置GPU import torch.nn.functional as F import torch.nn as nn import torch, torchvisiondevice torch.device("cuda"…

线程怎么创建?Java 四种方式一网打尽

&#x1f680; Java 中线程的 4 种创建方式详解 创建方式实现方式是否推荐场景说明1. 继承 Thread 类class MyThread extends Thread❌ 不推荐简单学习、单线程场景2. 实现 Runnable 接口class MyRunnable implements Runnable✅ 推荐更适合多线程共享资源3. 实现 Callable 接…