薛磊 Job Seeker

23种设计模式——20.Memento 备忘录(行为型模式)


动机(Motivation)

  在软件构建过程中,某些对象的状态在转换过程中,可能由于某种需要,要求程序能够回溯到对象之前处于某个点时的状态。如果使用一些公有接口来让其他对象得到对象的状态,便会暴露对象的细节实现。   如何实现对象状态的良好保存与恢复?但同时又不会因此而破坏对象本身的封装性。

意图(Intent)

  在不破坏封装性的前提下,捕获一个对象的内部状态,并在该对象之外保存这个状态。这样以后就可以将对象恢复到原先保存的状态。

结构(Structure)

  • Originator 发起人角色
  • Memento 备忘录角色
  • Caretaker 备忘录管理员角色

Memento模式的几个要点

  • 备忘录(Memento)存储原发器(Originator)对象的内部状态,在需要时恢复原发器状态。Memento模式适用于“由原发器管理,却又必须存储在原发器之外的信息”。

  • 在实现Memento模式中,要防止原发器以外的对象访问备忘录对象。备忘录对象有两个接口,一个为原发器使用的宽街口;一个为其他对象使用的窄接口。

  • 在实现Memento模式时,要考虑拷贝对象状态的效率问题,如果对象开销比较大,可以采用某种增量式改变来改变Memento模式。

示例代码

发起人角色:

public class Originator{
	//内部状态
	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 restoreMemento(Memento memento){
		this.setState(memento.getState());
	}
}

备忘录角色:

public class Memento{
	//发起人的内部状态
	private String state = "";
	//构造函数传递参数
	public Memento(String state){
		this.state = state;
	}
	public String getState(){
		return state;
	}
	public void setState(String state){
		this.state = state;
	}
}

备忘录管理员角色:

public class Caretaker{
	//备忘录对象
	private Memento memento;
	public Memento getMemento(){
		return memento;
	}
	public void setMemento(Memento memento){
		this.memento = memento;
	}
}

场景类:

public class Client{
	public static void main(String[] args){
		//定义出发起人
		Originator originator = new Originator();
		//定义出备忘录管理员
		Caretaker caretaker = new Caretaker();
		//创建一个备忘录
		caretaker.setMemento(originator.createMemento());
		//恢复一个备忘录
		originator.restoreMemento(caretaker.getMemento());
	}
}

发起人自主备份和恢复

public class Originator implements Cloneable{
	private Originator backup;
	//内部状态
	private String state = "";
	public String getState(){
		return state;
	}
	public void setState(String state){
		this.state = state;
	}
	//恢复一个备忘录
	public void restoreMemento(){
		//在进行恢复前应该进行断言,防止空指针
		this.setState(this.backup.getState());
	}
	//克隆当前对象
	protected Originator clone(){
		try{
			return (Originator)super.clone();
		} catch(CloneNotSupportedException e) {
			e.printStackTrace();
		}
		return null;
	}
}

Comments

Content