【设计模式】用Java实现备忘录模式

简介: 备忘录模式(Memento Pattern)是一种行为设计模式,它允许将对象的内部状态保存在一个备忘录对象中,并在需要时恢复对象的状态,而不破坏对象的封装性。

一.备忘录模式介绍与使用场景


备忘录模式(Memento Pattern)是一种行为设计模式,它允许将对象的内部状态保存在一个备忘录对象中,并在需要时恢复对象的状态,而不破坏对象的封装性。


备忘录模式的核心是备忘录(Memento)角色,它用于存储对象的内部状态,并提供对状态的访问方法。原发器(Originator)角色负责创建备忘录并将自身的状态保存到备忘录中。管理者(Caretaker)角色负责存储和恢复备忘录,但是它不能访问备忘录的内部状态。


应用场景:


1.当需要保存和恢复对象的内部状态,并且希望封装状态的具体实现细节时,可以使用备忘录模式。备忘录模式将对象的状态保存在备忘录对象中,对外部对象隐藏了状态的实现细节,提高了对象的封装性。

2.当需要在不破坏对象封装性的前提下保存和恢复对象的状态,以便进行撤销、回滚或者历史记录管理时,可以考虑使用备忘录模式。备忘录模式可以保存对象的历史状态,并在需要时恢复到指定的状态。

3.当需要保存对象状态的快照,以便在将来某个时间点恢复到该状态时,可以使用备忘录模式。备忘录模式可以捕获对象的当前状态并保存在备忘录对象中,以供后续恢复使用。

4.当对象的状态变化频繁,但是希望能够随时回滚到之前的某个状态时,可以考虑使用备忘录模式。备忘录模式可以在每次状态变化时保存备忘录,以便在需要时进行状态的回滚。

5.当需要实现多级撤销操作或者复杂的历史记录功能时,可以使用备忘录模式。备忘录模式可以保存多个备忘录对象,实现多级撤销或者浏览历史记录的功能。


需要注意的是,备忘录模式可能会占用较多的内存,特别是在需要保存大量状态或者频繁保存状态的情况下。因此,在使用备忘录模式时需要考虑内存的使用情况和性能要求。


二.备忘录模式实现


下面是一个使用Java实现的简单示例:

首先,我们定义备忘录类 Memento,用于存储对象的状态:

public class Memento {
    private String state;
    public Memento(String state) {
        this.state = state;
    }
    public String getState() {
        return state;
    }
}

然后,我们创建原发器类 Originator,它包含了需要保存和恢复的状态,并提供了创建备忘录和从备忘录恢复状态的方法:

public class Originator {
    private String state;
    public void setState(String state) {
        this.state = state;
    }
    public String getState() {
        return state;
    }
    public Memento createMemento() {
        return new Memento(state);
    }
    public void restoreFromMemento(Memento memento) {
        state = memento.getState();
    }
}

接下来,我们创建管理者类 Caretaker,它负责存储和恢复备忘录对象:

public class Caretaker {
    private Memento memento;
    public void setMemento(Memento memento) {
        this.memento = memento;
    }
    public Memento getMemento() {
        return memento;
    }
}

最后,我们可以在客户端中使用备忘录模式:

public class Client {
    public static void main(String[] args) {
        Originator originator = new Originator();
        Caretaker caretaker = new Caretaker();
        // 设置原发器的状态
        originator.setState("State 1");
        System.out.println("Initial state: " + originator.getState());
        // 创建备忘录并保存到管理者
        caretaker.setMemento(originator.createMemento());
        // 修改原发器的状态
        originator.setState("State 2");
        System.out.println("Updated state: " + originator.getState());
        // 从备忘录中恢复原发器的状态
        originator.restoreFromMemento(caretaker.getMemento());
        System.out.println("Restored state: " + originator.getState());
    }
}

在上面的示例中,我们创建了一个原发器对象 originator 和一个管理者对象 caretaker。首先,我们设置原发器的初始状态并打印出来。然后,创建备忘录并将其保存到管理者中。接下来,修改原发器的状态并打印出来。最后,从备忘录中恢复原发器的状态并打印出来,可以看到原发器的状态被成功恢复。


通过使用备忘录模式,我们可以在不破坏对象封装的前提下保存和恢复对象的状态。这对于需要记录和回滚对象状态的场景非常有用,例如撤销操作、快照管理等。


下面再举一个实际项目环境中的例子来说明


假设我们正在开发一个文本编辑器,用户可以在编辑器中输入文本并进行编辑操作。我们希望使用备忘录模式来实现撤销和恢复功能,即用户可以撤销之前的编辑操作,并恢复到之前的状态。


首先,我们定义备忘录类 EditorMemento,用于保存编辑器的状态:

public class EditorMemento {
    private String content;
    public EditorMemento(String content) {
        this.content = content;
    }
    public String getContent() {
        return content;
    }
}

然后,我们创建原发器类 Editor,它包含需要保存和恢复的状态,并提供了创建备忘录和从备忘录恢复状态的方法:

public class Editor {
    private String content;
    public void setContent(String content) {
        this.content = content;
    }
    public String getContent() {
        return content;
    }
    public EditorMemento createMemento() {
        return new EditorMemento(content);
    }
    public void restoreFromMemento(EditorMemento memento) {
        content = memento.getContent();
    }
}

接下来,我们创建管理者类 History,它负责存储和恢复备忘录对象:

import java.util.Stack;
public class History {
    private Stack<EditorMemento> mementos;
    public History() {
        mementos = new Stack<>();
    }
    public void pushMemento(EditorMemento memento) {
        mementos.push(memento);
    }
    public EditorMemento popMemento() {
        return mementos.pop();
    }
}

最后,我们可以在客户端中使用备忘录模式:

public class Client {
    public static void main(String[] args) {
        Editor editor = new Editor();
        History history = new History();
        // 用户输入和编辑文本
        editor.setContent("First version");
        System.out.println("Current content: " + editor.getContent());
        // 创建备忘录并保存到历史记录
        history.pushMemento(editor.createMemento());
        // 用户继续编辑文本
        editor.setContent("Second version");
        System.out.println("Current content: " + editor.getContent());
        // 用户撤销操作,恢复到之前的状态
        EditorMemento memento = history.popMemento();
        editor.restoreFromMemento(memento);
        System.out.println("Restored content: " + editor.getContent());
    }
}

在上面的示例中,我们创建了一个原发器对象 editor 和一个历史记录对象 history。首先,用户输入和编辑了文本,并打印出当前的内容。然后,创建备忘录并将其保存到历史记录中。接下来,用户继续编辑文本并打印出当前内容。最后,用户撤销操作,从历史记录中取出备忘录并恢复到之前的状态,然后打印出恢复后的内容。


相关文章
|
6天前
|
设计模式 Java 关系型数据库
【Java笔记+踩坑汇总】Java基础+JavaWeb+SSM+SpringBoot+SpringCloud+瑞吉外卖/谷粒商城/学成在线+设计模式+面试题汇总+性能调优/架构设计+源码解析
本文是“Java学习路线”专栏的导航文章,目标是为Java初学者和初中高级工程师提供一套完整的Java学习路线。
|
2天前
|
设计模式 Java
Java设计模式:组合模式的介绍及代码演示
组合模式是一种结构型设计模式,用于将多个对象组织成树形结构,并统一处理所有对象。例如,统计公司总人数时,可先统计各部门人数再求和。该模式包括一个通用接口、表示节点的类及其实现类。通过树形结构和节点的通用方法,组合模式使程序更易扩展和维护。
Java设计模式:组合模式的介绍及代码演示
|
6天前
|
设计模式 安全 算法
【Java面试题汇总】设计模式篇(2023版)
谈谈你对设计模式的理解、七大原则、单例模式、工厂模式、代理模式、模板模式、观察者模式、JDK中用到的设计模式、Spring中用到的设计模式
【Java面试题汇总】设计模式篇(2023版)
|
6天前
|
设计模式 Java 关系型数据库
【Java笔记+踩坑】设计模式——原型模式
对比原型模式和传统方式的实现思路、代码方案、优缺点,阐述原型模式的使用场景,以及深拷贝、浅拷贝等相关概念,并扩展原型模式在Spring源码中的应用。
【Java笔记+踩坑】设计模式——原型模式
|
21天前
|
设计模式 缓存 算法
揭秘策略模式:如何用Java设计模式轻松切换算法?
【8月更文挑战第30天】设计模式是解决软件开发中特定问题的可重用方案。其中,策略模式是一种常用的行为型模式,允许在运行时选择算法行为。它通过定义一系列可互换的算法来封装具体的实现,使算法的变化与客户端分离。例如,在电商系统中,可以通过定义 `DiscountStrategy` 接口和多种折扣策略类(如 `FidelityDiscount`、`BulkDiscount` 和 `NoDiscount`),在运行时动态切换不同的折扣逻辑。这样,`ShoppingCart` 类无需关心具体折扣计算细节,只需设置不同的策略即可实现灵活的价格计算,符合开闭原则并提高代码的可维护性和扩展性。
37 2
|
21天前
|
设计模式 Java
Java 设计模式之谜:工厂模式与抽象工厂模式究竟隐藏着怎样的神奇力量?
【8月更文挑战第30天】在Java编程中,设计模式为常见问题提供了高效解决方案。工厂模式与抽象工厂模式是常用的对象创建型设计模式,能显著提升代码的灵活性、可维护性和可扩展性。工厂模式通过定义创建对象的接口让子类决定实例化哪个类;而抽象工厂模式则进一步提供了一个创建一系列相关或相互依赖对象的接口,无需指定具体类。这种方式使得系统更易于扩展和维护。
30 1
|
21天前
|
设计模式 Java
重构你的代码:探索Java中的混合、装饰器与组合设计模式
【8月更文挑战第30天】在软件开发中,设计模式为特定问题提供了结构化的解决方案,使代码更易理解、维护及扩展。本文将介绍三种常用的 Java 设计模式:混合模式、装饰器模式与组合模式,并附有示例代码展示实际应用。混合模式允许通过继承多个接口或抽象类实现多重继承;装饰器模式可在不改变对象结构的情况下动态添加新功能;组合模式则通过树形结构表示部分-整体层次,确保客户端处理单个对象与组合对象时具有一致性。
15 1
|
6天前
|
设计模式 算法 安全
设计模式——模板模式
模板方法模式、钩子方法、Spring源码AbstractApplicationContext类用到的模板方法
设计模式——模板模式
|
1月前
|
设计模式
设计模式-单一职责模式
设计模式-单一职责模式
|
1月前
|
设计模式 XML 存储
【二】设计模式~~~创建型模式~~~工厂方法模式(Java)
文章详细介绍了工厂方法模式(Factory Method Pattern),这是一种创建型设计模式,用于将对象的创建过程委托给多个工厂子类中的某一个,以实现对象创建的封装和扩展性。文章通过日志记录器的实例,展示了工厂方法模式的结构、角色、时序图、代码实现、优点、缺点以及适用环境,并探讨了如何通过配置文件和Java反射机制实现工厂的动态创建。
【二】设计模式~~~创建型模式~~~工厂方法模式(Java)