【设计模式】备忘录模式 ( 简介 | 适用场景 | 优缺点 | 代码示例 )





一、备忘录模式简介



备忘录模式 : 保存 对象 某个状态 , 以便在 适当的时候 恢复对象 ;
( 形象的比喻 : " 后悔药 " )

如 : 游戏存档 , 一些编辑工具中的 " 撤销 " 操作 , 浏览器中的 后退 ;


备忘录模式 类型 : 行为型 ;





二、备忘录模式 适用场景



备忘录模式 适用场景 :

  • 撤销操作 : 保存 / 恢复 数据 的相关业务场景 ;
    如 : 在 Word 中编写文档 , 如果想要撤销之前的 输入 / 删除操作 , 使用 Ctrl + Z 执行 " 撤销 " 操作 ;

  • 状态恢复 : 在 " 后悔 " 的时候 , 将对象恢复到之前的状态 ;
    如 : 游戏中的存档使用 ;





三、备忘录模式 优缺点



备忘录模式 优点 :

  • 存档信息 : 封装 存档信息 ;
  • 恢复机制 : 为 用户 提供一种 可恢复 机制 ;

先封装 存档信息 , 然后才可以提供 可恢复机制 ;
封装的 对象的状态 , 就是 对象中 各个属性的属性值 , 快照 ;


备忘录模式 缺点 : 资源占用 , 会额外 占用 磁盘 / 内存 等资源 ;





四、备忘录模式 与 状态模式



备忘录模式 与 状态模式 :

  • 备忘录模式状态表示 : 备忘录模式 中 , 使用 对象实例 表示状态 , 当前对象的 存档 是该对象的实例 ;

  • 状态模式状态表示 : 状态模式 中 , 使用 表示状态 ;





五、备忘录模式 代码示例



业务场景 : 编辑文档 , 有暂存功能 , 暂时先保存到内存中 ;


1、文档类


package memento;

/**
 * 文档
 *      需要保存的对象
 */
public class Article {
    private String title;
    private String content;
    private String image;

    public Article(String tittle, String content, String image) {
        this.title = tittle;
        this.content = content;
        this.image = image;
    }

    public String getTitle() {
        return title;
    }

    public void setTitle(String title) {
        this.title = title;
    }

    public String getContent() {
        return content;
    }

    public void setContent(String content) {
        this.content = content;
    }

    public String getImage() {
        return image;
    }

    public void setImage(String image) {
        this.image = image;
    }

    /**
     * 保存信息到备忘录
     * @return
     */
    public ArticleMemento saveToMemento() {
        ArticleMemento articleMemento = new ArticleMemento(title, content, image);
        return articleMemento;
    }

    /**
     * 从备忘录恢复
     * @param articleMemento
     */
    public void undoFromMemento(ArticleMemento articleMemento) {
        this.title = articleMemento.getTitle();
        this.content = articleMemento.getContent();
        this.image = articleMemento.getImage();
    }

    @Override
    public String toString() {
        return "Article{" +
                "title='" + title + '\'' +
                ", content='" + content + '\'' +
                ", image='" + image + '\'' +
                '}';
    }
}

2、文档备忘录类


package memento;

/**
 * 文档的备忘录类
 *      主要用于存储文档的各种属性状态信息
 *  备忘录 快照 没有 set 方法
 *      只能通过构造函数设置备忘录数据
 */
public class ArticleMemento {
    private String title;
    private String content;
    private String image;

    public ArticleMemento(String title, String content, String image) {
        this.title = title;
        this.content = content;
        this.image = image;
    }

    public String getTitle() {
        return title;
    }

    public String getContent() {
        return content;
    }

    public String getImage() {
        return image;
    }

    @Override
    public String toString() {
        return "ArticleMemento{" +
                "title='" + title + '\'' +
                ", content='" + content + '\'' +
                ", image='" + image + '\'' +
                '}';
    }
}

3、文档备忘录管理类


package memento;

import java.util.Stack;

/**
 * 备忘录管理类
 */
public class ArticleMementoManager {
    /**
     * 存储所有的备忘录信息
     *      在 栈 数据结构中存储 , 特点后进先出
     */
    private final Stack<ArticleMemento> mArticleMementoStack = new Stack<>();

    /**
     * 获取栈顶的备忘录信息
     * @return
     */
    public ArticleMemento getArticleMemento() {
        return mArticleMementoStack.pop();
    }

    /**
     * 备忘录信息入栈
     *      放在栈顶
     * @param articleMemento
     */
    public void setArticleMemento(ArticleMemento articleMemento) {
        mArticleMementoStack.push(articleMemento);
    }
}

4、测试类


package memento;

public class Main {
    public static void main(String[] args) {
        ArticleMementoManager articleMementoManager = new ArticleMementoManager();

        // 创建并输入文档内容
        Article article = new Article("标题", "内容", "图片链接");
        // 保存备忘录信息
        ArticleMemento articleMemento = article.saveToMemento();
        // 将备忘录信息设置到 备忘录管理者
        articleMementoManager.setArticleMemento(articleMemento);
        // 打印备忘录内容
        System.out.println("文档信息 : " + article.toString());

        // 修改文档内容
        article.setTitle("标题 2");
        article.setContent("内容 2");
        article.setImage("图片链接 2");
        // 保存新的备忘录信息
        articleMemento = article.saveToMemento();
        // 将备忘录信息设置到 备忘录管理者
        articleMementoManager.setArticleMemento(articleMemento);
        // 打印备忘录内容
        System.out.println("文档信息 : " + article.toString());

        // 此时 ArticleMementoManager 中存储了 2 个存档
        // 存档 1 : Article{title='标题', content='内容', image='图片链接'}
        // 存档 2 : Article{title='标题 2', content='内容 2', image='图片链接 2'}

        // 使用备忘录回退
        // 先将栈顶的当前备忘录出栈 , 移除
        articleMementoManager.getArticleMemento();
        // 然后获取上一个备忘录 , 并设置到 Article 中
        article.undoFromMemento(articleMementoManager.getArticleMemento());
        // 打印备忘录内容
        System.out.println("文档信息 : " + article.toString());
    }
}
  • 4
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
设计模式之备忘录 和 状态模式精讲 19.1 场景问题 19.1.1 开发仿真系统 考虑这样一个仿真应用,功能是:模拟运行针对某个具体问题的多个解决方案,记录运行过程的各种数据,在模拟运行完成过后,好对这多个解决方案进行比较和评价,从而选定最优的解决方案。 这种仿真系统,在很多领域都有应用,比如:工作流系统,对同一问题制定多个流程,然后通过仿真运行,最后来确定最优的流程做为解决方案;在工业设计和制造领域,仿真系统的应用就更广泛了。 由于都是解决同一个具体的问题,这多个解决方案并不是完全不一样的,假定它们的前半部分运行是完全一样的,只是在后半部分采用了不同的解决方案,后半部分需要使用前半部分运行所产生的数据。 由于要模拟运行多个解决方案,而且最后要根据运行结果来进行评价,这就意味着每个方案的后半部分的初始数据应该是一样,也就是说在运行每个方案后半部分之前,要保证数据都是由前半部分运行所产生的数据,当然,咱们这里并不具体的去深入到底有哪些解决方案,也不去深入到底有哪些状态数据,这里只是示意一下。 那么,这样的系统该如何实现呢?尤其是每个方案运行需要的初始数据应该一样,要如何来保证呢? 19.1.2 不用模式的解决方案 要保证初始数据的一致,实现思路也很简单: 首先模拟运行流程第一个阶段,得到后阶段各个方案运行需要的数据,并把数据保存下来,以备后用 每次在模拟运行某一个方案之前,用保存的数据去重新设置模拟运行流程的对象,这样运行后面不同的方案时,对于这些方案,初始数据就是一样的了 根据上面的思路,来写出仿真运行的示意代码示例代码如下: /** * 模拟运行流程A,只是一个示意,代指某个具体流程 */ public class FlowAMock { /** * 流程名称,不需要外部存储的状态数据 */ private String flowName; /** * 示意,代指某个中间结果,需要外部存储的状态数据 */ private int tempResult; /** * 示意,代指某个中间结果,需要外部存储的状态数据 */ private String tempState; /** * 构造方法,传入流程名称 * @param flowName 流程名称 */ public FlowAMock(String flowName){ this.flowName = flowName; } public String getTempState() { return tempState; } public void setTempState(String tempState) { this.tempState = tempState; } public int getTempResult() { return tempResult; } public void setTempResult(int tempResult) { this.tempResult = tempResult; } /** * 示意,运行流程的第一个阶段 */ public void runPhaseOne(){ //在这个阶段,可能产生了中间结果,示意一下 tempResult = 3; tempState = "PhaseOne"; } /** * 示意,按照方案一来运行流程后半部分 */ public void schema1(){ //示意,需要使用第一个阶段产生的数据 this.tempState += ",Schema1"; System.out.println(this.tempState + " : now run "+tempResult); this.tempResult += 11; } /** * 示意,按照方案二来运行流程后半部分 */ public void schema2(){ //示意,需要使用第一个阶段产生的数据 this.tempState += ",Schema2"; System.out.println(this.tempState + " : now run "+tempResult); this.tempResult += 22; } } (2)看看如何使用这个模拟流程的对象,写个客户端来测试一下。示例代码如下: public class Client { public static void main(String[] args) { // 创建模拟运行流程的对象 FlowAMock mock = new FlowAMock("TestFlow"); //运行流程的第一个阶段 mock.runPhaseOne(); //得到第一个阶段运行所产生的数据,后面要用 int tempResult = mock.getTempResult(); String tempState = mock.getTempState(); //按照方案一来运行流程后半部分 mock.schema1(); //把第一个阶段运行所产生的数据重新设置回去 mock.setTempResult(tempResult); mock.setTempState(tempState); //按照方案二来运行流程后半部分 mock.schema2(); } } 运行结果如下: PhaseOne,Schema1 : now run 3 PhaseOne,Schema2 : now run 3
备忘录模式是一种行为型设计模式,它允许在不破坏封装性的前提下捕获并保存一个对象的内部状态,并在需要时恢复该状态。在软件开发中,备忘录模式通常用于实现撤销操作或者历史记录功能。 下面是一个使用备忘录模式的简单示例,假设我们有一个文本编辑器,用户可以在其中输入文本并进行编辑操作。我们希望实现一个撤销功能,使得用户可以撤销之前的编辑操作。 首先,我们需要定义一个备忘录类,用于保存编辑器的状态: ```python class EditorMemento: def __init__(self, content): self.content = content def get_content(self): return self.content ``` 然后,我们需要定义一个编辑器类,其中包含了一些编辑操作,以及用于保存和恢复状态的方法: ```python class Editor: def __init__(self): self.content = "" def type(self, words): self.content = self.content + " " + words def delete(self, words): if words in self.content: self.content = self.content.replace(words, "") def save(self): return EditorMemento(self.content) def restore(self, memento): self.content = memento.get_content() ``` 最后,我们可以使用这个编辑器类来实现撤销功能: ```python editor = Editor() editor.type("This is the first sentence.") editor.type("This is the second sentence.") saved = editor.save() editor.delete("second") print(editor.content) # 输出:This is the first sentence. editor.restore(saved) print(editor.content) # 输出:This is the first sentence. This is the second sentence. ```

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值