19.java设计模式之备忘录模式
基本需求
- 游戏的角色有攻击力和防御力,在大战Boss之前保存自身的状态(攻击力和防御力),当大战Boss之后攻击力和防御力下降,从备忘录对象恢复到大战前的状态
传统方案
-
一个对象,就对应一个保存对象状态的对象
-
说明
- 一个对象,就对应一个保存对象状态的对象,这样当我们游戏的对象很多时,不利于管理,开销也很大
- 传统的方式是简单的做备份,new出来另外一个对象出来,再把需要备份的数据放到这个新对象,但这就暴露了对象内部的细节
- 可以使用备忘录模式进行解决
基本介绍
-
备忘录模式(Memento)在 不破坏封装性的前提下,捕获 一个对象的内部状态,并在该对象之外保存这个状态。这样以后就可将该对象恢复到原先保存的状态
-
可以这里理解备忘录模式:现实生活中的备忘录是用来记录某些要去做的事情,或者是记录已经达成的共同意见的事情,以防忘记了。而在软件层面,备忘录模式有着相同的含义,备忘录对象主要用来记录一个对象的某
种状态,或者某些数据,当要做回退时,可以从备忘录对象里获取原来的数据进行恢复操作 -
备忘录模式属于行为型模式
-
UML类图(原理)
-
说明
- Originator:对象,需要保存状态的对象
- Memento:备忘录对象,保存目标对象Originator的内部状态
- Caretaker:守护者对象,保存多个备忘录对象,使用集合管理,提高效率
- 如果希望保存多个Originator对象不同时间的状态,可使用Map<String,List
>进行管理
-
代码实现
-
// 备忘录对象 @Data @NoArgsConstructor @AllArgsConstructor public class Memento { private String state; } // 目标对象 @Data public class Originator { private String state; // 创建备忘录对象,用户备份 public Memento createMemento() { return new Memento(this.state); } // 从备忘录对象恢复对象 public void restoreFromMemento(Memento memento) { this.state = memento.getState(); } } // 守护者对象 public class Caretaker { // 使用集合对备忘录对象进行管理 private List<Memento> mementos; public Caretaker() { this.mementos = new ArrayList<>(); } // 添加备忘录对象 public void add(Memento memento) { mementos.add(memento); } // 根据索引在集合中获取备忘录对象 public Memento getMementoByIndex(int index) { Memento memento = null; if (index >= 0 && index < mementos.size()) { memento = mementos.get(index); } return memento; } } // 测试 public class Client { public static void main(String[] args) { // 创建守护者对象 Caretaker caretaker = new Caretaker(); // 创建目标对象 并设置状态1 Originator originator = new Originator(); originator.setState("状态1 -> 开心"); // 保存状态1 -> 获取备忘录对象,并交由守护者管理 caretaker.add(originator.createMemento()); // 给目标状态设置状态2 originator.setState("状态2 -> 悲伤"); // 保存状态2 -> 获取备忘录对象,并交由守护者管理 caretaker.add(originator.createMemento()); // 给目标状态设置状态3 originator.setState("状态3 -> 笑哭"); // 保存状态3 -> 获取备忘录对象,并交由守护者管理 caretaker.add(originator.createMemento()); // 目标对象当前的状态 System.out.println("目标对象当前的状态是:" + originator.getState()); // 从守护者对象中获取备忘录对象,恢复目标对象状态至状态1 originator.restoreFromMemento(caretaker.getMementoByIndex(0)); System.out.println("目标对象状态恢复到状态1:" + originator.getState()); } }
-
-
UML类图(案例)
-
代码实现
-
// 备忘录对象 @Data @NoArgsConstructor @AllArgsConstructor public class Memento { // 攻击力 private int attack; // 防御力 private int defense; } // 游戏角色 亦目标对象 @Data public class GameRole { // 攻击力 private int attack; // 防御力 private int defense; // 创建目标对象当前状态的备忘录对象 public Memento createMemento() { return new Memento(this.attack, this.defense); } // 从备忘录对象中恢复目标对象 public void restoreFromMemento(Memento memento) { this.attack = memento.getAttack(); this.defense = memento.getDefense(); } // 显示目标对象当前状态 public void display() { System.out.println("目标对象当前状态,攻击力 -> " + this.attack + "、防御力 -> " + this.defense); } } // 守护者对象 对备忘录对象进行管理 @Data public class Caretaker { // 只保存一次状态 根据本次需求 只需保存一次 private Memento memento; // 保存一个目标对象的多次状态 // private List<Memento> mementos; // 保存多个目标对象的多次状态 // private Map<String, List<Memento>> listMap; } // 测试 public class Client { public static void main(String[] args) { // 创建守护者对象 管理备忘录对象 Caretaker caretaker = new Caretaker(); // 创建目标对象 并设置初始值 GameRole gameRole = new GameRole(); gameRole.setAttack(100); gameRole.setDefense(100); // 获取目标对象当前状态的备忘录对象,并保存至守护者对象中 caretaker.setMemento(gameRole.createMemento()); // 和boss大战前 输出目标对象状态 System.out.println("和boss大战前"); gameRole.display(); // 和boss大战中 输出目标对象状态 System.out.println("和boss大战中"); gameRole.setAttack(50); gameRole.setDefense(50); gameRole.display(); // 和boss大战后 使用备忘录对象恢复目标对象状态至初始状态 gameRole.restoreFromMemento(caretaker.getMemento()); System.out.println("和boss大战后,恢复至初始状态"); gameRole.display(); } }
-
注意事项
- 给用户提供了一种可以恢复状态的机制,可以使用户能够比较方便的回到某个历史状态
- 实现了信息封装,使得用户不需要关心状态的保存细节
- 如果类的成员变量过多,势必会占用比较大的资源,而且每一次保存都会消耗一定的内存
- 应用场景:1、后悔药,2、游戏存档,3、ctrl+z,4、ie中的后退,5、数据库的事务管理
- 为了节约内存,备忘录模式可以和原型模式配合使用