18.备忘录模式(Memento Pattern)

阅读更多

引子

俗话说:世上难买后悔药。所以凡事讲究个“三思而后行”,但总常见有人做“痛心疾首”状:当初我要是……。如果真的有《大话西游》中能时光倒流的“月光宝盒”,那这世上也许会少一些伤感与后悔——当然这只能是痴人说梦了。

但是在我们手指下的程序世界里,却有的后悔药买。今天我们要讲的备忘录模式便是程序世界里的“月光宝盒”。

所谓备忘录模式就是在执行某个命令之前先将当前状态备份,执行完,在某种情况下需要将状态回滚。

 

1.定义

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

 

2.备忘录模式的三个角色

  • Originator发起人角色:记录当前时刻的内部状态,负责定义哪些属于备份范围的状态,负责创建和恢复备忘录数据。
  • Memento备忘录角色:负责存储Originator发起人对象的内部状态,在需要的时候提供发起人需要内部状态。
  • Caretaker备忘录管理员角色:对备忘录进行管理、保存和提供备忘录。

3.备忘录模式的使用场景

需要保存和恢复数据的相关场景
提供一个可回滚(rollback)的操作;比如CTRL+Z组合键,IE浏览器的后退按钮等
需要监控的副本场景中:例如需要监控一个对象的属性,但是监控又不应该作为系统的主要业务来调用,它只是边缘应用,即使出现监控不准,错误报警也影响不大,因此一般的做法是备份一个主线程中的对象,然后由分析程序来分析
数据库连接的事物管理就是一个备忘录模式,想想看,如果你要实现一个JDBC驱动,如何来实现事物?还不是使用备忘录模式!

 

下面是备忘录模式的通用代码:

 

package _18MementoPattern;

/**
 * 备忘录角色
 */
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;
	}
}
 
package _18MementoPattern;

/**
 * 发起人角色
 */
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(state);
	}
	
	// 使用备忘录恢复状态
	public void recover(Memento memento)
	{
		this.state = memento.getState();
	}
}
 
package _18MementoPattern;

/**
 * 备忘录管理员
 */
public class Caretaker {

	// 管理员负责管理备忘录
	private Memento memento;

	public Memento getMemento() {
		return memento;
	}

	public void setMemento(Memento memento) {
		this.memento = memento;
	}
}
 
package _18MementoPattern;

/**
 * 场景类
 */
public class Client {

	public static void main(String[] args) {
		// 初始状态
		Originator originator = new Originator();
		originator.setState("开心");
		System.out.println("初始状态:" + originator.getState());
		// 备份初始状态
		Caretaker caretaker = new Caretaker();
		caretaker.setMemento(originator.createMemento());
		// 表白被拒绝了
		originator.setState("悲伤");
		System.out.println("表白被拒:" + originator.getState());
		// 恢复状态
		originator.recover(caretaker.getMemento());
		System.out.println("恢复状态:" + originator.getState());
	}

}

 

4.备忘录模式的注意事项

  • 备忘录的生命周期:备忘录创建出来就要在“最近”的代码中使用,要主动管理它的生命周期,不使用时删除其引用,等待垃圾回收。
  • 备忘录的性能:不要在频繁建立备份的场景中使用备忘录模式(比如for循环),原因有二:一是控制不了备忘录建立的对象数量,二是大对象的建立是要消耗资源的,系统的性能需要考虑。

5.备忘录模式的扩展
5.1 clone方式的备忘录

当然备忘录模式也可以灵活变化,比如发起人融合了备忘录角色:

 

package _18MementoPattern;

/**
 * 发起人角色融合了备忘录角色
 */
public class Originator2 implements Cloneable {

	// 发起人的内部状态
	private String state = "";

	public String getState() {
		return state;
	}

	public void setState(String state) {
		this.state = state;
	}
	
	// 创建一个备忘录
	public Originator2 createMemento()
	{
		return this.clone();
	}
	
	// 使用备忘录恢复状态
	public void recover(Originator2 originator2)
	{
		this.state = originator2.getState();
	}

	@Override
	protected Originator2 clone(){
		try {
			return (Originator2)super.clone();
		} catch (CloneNotSupportedException e) {
			e.printStackTrace();
		}
		return null;
	}
}

 

还可以更简单,发起人不仅融合了备忘录角色,还融合了备忘录管理员角色:

 

package _18MementoPattern;

/**
 * 发起人角色同事融合了备忘录角色和管理员角色
 */
public class Originator3 implements Cloneable {
	
	// 备忘录
	private Originator3 backup;

	// 发起人的内部状态
	private String state = "";

	public String getState() {
		return state;
	}

	public void setState(String state) {
		this.state = state;
	}
	
	// 创建一个备忘录
	public void createMemento()
	{
		backup = this.clone();
	}
	
	// 使用备忘录恢复状态
	public void recover()
	{
		this.state = backup.getState();
	}

	@Override
	protected Originator3 clone(){
		try {
			return (Originator3)super.clone();
		} catch (CloneNotSupportedException e) {
			e.printStackTrace();
		}
		return null;
	}
}

 

5.2 多状态的备忘录模式
5.3 多备份的备忘录模式

你可能感兴趣的:(备忘录模式,Memento,Pattern)