有时候我们有必要记录一个对象的内部状态,可以让用户取消不确定的操作或从错误中恢复过来,但是又不能破坏封装性。我们可用备忘录模式,一个备忘录是一个对象,它存储另一个对象在某个瞬间的内部状态,而后者称为备忘录的原发器。当需要设置原发器的检查点时,取消操作机制会向原发器请求一个备忘录。原发器用描述当前状态的信息初始化该备忘录,只有原发器可以向备忘录存取信息,备忘录对其他对象不可见。
其适用性:
必须保存一个对象在某一个时刻的(部分) 状态,这样以后需要时它才能恢复到先前的状态,
如果一个用接口来让其他对象直接得到这些状态,将会暴露对象的实现细节并破坏对象的封装性。
其结构图:
Memento来负责备忘录存储原发器对象的内部状态,其实际上有两个接口,管理者只能看到备忘录的窄接口,它只能将备忘录传递给其他对象。相反,原发器能够看到一个宽接口,允许它访问返回到先前状态所需的所有数据。Originator负责创建备忘录,Caretaker负责保存备忘录。实现如下:
package org.designpattern.behavioral.memento;
private String state;
public Memento(String state){
this.state = state;
}
public String getState() {
return state;
}
}
原发器创建备忘录,对它的状态进行赋值:
package org.designpattern.behavioral.memento;
private String state;
public String getState() {
return state;
}
public void setState(String state) {
this.state = state;
}
public Memento createMemento(){
return new Memento(this.state);
}
public void setStateFromMemento(Memento memento){
this.state = memento.getState();
}
}
管理器向原发器请求一个备忘录,并保存下来,需要时送回原发器:
package org.designpattern.behavioral.memento;
private Memento memento;
public Memento getMemento() {
return memento;
}
public void setMemento(Memento memento) {
this.memento = memento;
}
}
下面是客户段测试类:
package org.designpattern.behavioral.memento;
public static void main(String[] args) {
Originator originator = new Originator();
originator.setState("loving");
Memento memento = originator.createMemento();
System.out.println(originator.getState());
Caretaker caretaker = new Caretaker();
caretaker.setMemento(memento);
originator.setState("divorce");
System.out.println(originator.getState());
originator.setStateFromMemento(caretaker.getMemento());
System.out.println(originator.getState());
}
}
该模式把可能复杂的Originator内部信息对其他对象屏蔽起来,如果原发器在生成备忘录时必须拷贝并存储大量的信息或者其他情况,有可能会导致非常大的开销。因为该模式的接口较小,以至于备忘录只能作为一个值传递。其常用来与Command模式一起用来维护可以撤销的操作。