设计模式之备忘录模式

原创
2019/12/26 17:51
阅读数 92

定义

备忘录模式(Momento Pattern):在不破坏封装性的前提下,捕获一个对象的内部状态,并在该对象之外保存这个状态,以便以后当需要时能将该对象恢复到原先保存的状态。又叫快照模式。

备忘录模式是一个比较简单的模式,用于备份对象状态。通常可以与命令模式等一起使用。

角色

备忘录模式有三个角色:备忘录(Memento)角色,发起人(Originator)角色,负责人(Caretaker)角色。

从类图上可以看出,发起人和备忘录角色都有 state 属性,其实可以把备忘录角色当作发起人角色的属性容器,发起人决定要备份哪些状态,并将状态放到备忘录中,备忘录存储这些状态,并且可以被保存起来,在需要的时候提供给发起人用于恢复到以前。

模式说明

我们结合命令模式来说明备忘录模式,依然使用浏览器的返回场景来说明。用户在浏览器中点击某个超链接跳到另一个页面,点击返回回到上一个链接页面,我们将点击抽象成命令,将返回抽象成恢复状态,浏览器就是我们的发起人角色。

/**
  * 在浏览器中是URL跳转
  */
public interface Command<T> {
    void execute(T param);
    void back();
}

@Data
public class Memento {
    private String url;
}
// 浏览器窗口
public class Window {
    private String url;
    
    public Memento createMemento() {
        Memento m = new Memento();
        m.setUrl(this.url);
        return m;
    }
    
    public void restoreMemento(Memento m) {
        rendUrlAndShow(m.getUrl());
    }
    
    public void rendUrlAndShow(String url) {
        this.url = url;
        System.out.println(url);
    }
}
public class UrlCommand implements Command<String> {
    private Window window;
    private Memento m;

    public void execute(String url) {
        this.m = window.createMemento();
        window.rendUrlAndShow(url);
    }
    
    public void back() {
        // 将原窗口再设置回去
        window.restoreMemento(m);
    }
}

public class User {
    public static void main(String[] args) {
        // 在浏览器里用户点击 a.com
        String url = "www.a.com";
        Command<String> commandA = new UrlCommand();
        commandA.execute(url);
        
        // 用户点击返回
        stack.pop().back();
    }
}

在这里 UrlCommand 既是 命令模式中的Command 角色也是备忘录模式中的 负责人角色。

模式变种

  1. 由于备忘录角色只是一个属性容器,用于保存需要的属性,因此发起人自己就可以充当备忘录角色,这时可以使原型模式来创建当前对象的克隆,然后保存这个副本即可

  2. 备忘录保存的是发起对象的属性,这些属性可以直接保存到 key-value 的 map 中,无需单独创建一个类

  3. 备忘录模式只是描述单一备份的情况,在有多个备份的情况下可以配合 Stack 等数据结构使用,例如浏览器支持多次返回,例子如下:

public class User {
    public static void main(String[] args) {
        Stack<String> stack = new Stack<>();
        // 在浏览器里用户点击 a.com
        String url = "www.a.com";
        Command<String> commandA = new UrlCommand();
        commandA.execute(url);
        stack.push(commandA);
        // 用户点击b.com
        url = "www.b.com"
        Command<String> commandB = new UrlCommand();
        commandB.execute(url);
        stack.push(commandB);
        
        // 用户点击返回
        stack.pop().back();
        // 用户又点击一次返回
        stack.pop().back();
    }
}

优点

  • 在不破坏封装的前提下保存一个对象的状态
  • 与其他模式共同使用,可以方便实现一些功能,如命令模式的 undo
  • 备忘录模式是实现检查点(check point)的一种方法

缺点

  • 在备份对象过多时,会占用很多内存,此时需要与外部存储交互
展开阅读全文
打赏
0
0 收藏
分享
加载中
更多评论
打赏
0 评论
0 收藏
0
分享
返回顶部
顶部