备忘录模式

2018-01-09 16:59:41 浏览数 (1)

备忘录模式,望文生义就知道它是用来做备忘的,或者可以直接说是“备份”。当需要保存当前状态,以便在不久要恢复此状态时,就可以使用“备忘录模式”。将当前”状态“备份,是不是又new一个类,然后将每个字段方法copy过去就可以了呢?或者说使用我们之前clone方法做深复制浅复制呢?其实不然,在《大话设计模式》中,作者提到了原因,这样会暴露更多的细节给客户端,不符合我们面向对象的思想。什么是暴露更多的细节给客户端?我们来看下面一段代码。

代码语言:javascript复制
 1 package day_27_memento;
 2 
 3 /**
 4  * @author turbo
 5  *
 6  * 2016年9月27日
 7  */
 8 public class Client {
 9 
10     /**
11      * @param args
12      */
13     public static void main(String[] args) {
14         /*状态一*/
15         Test nowTest = new Test();
16         nowTest.setField("状态一");
17         
18         /*备份状态*/
19         Test backUpTest = new Test();
20         backUpTest.setField(nowTest.getField());
21         
22         /*修改状态一*/
23         nowTest.setField("修改状态一,改为状态二");
24         
25         /*还原状态*/
26         nowTest.setField(backUpTest.getField());
27     }
28 
29 }

Test类中只有一个field字段,不再贴出代码。由上代码我们可以看到,如果要备份的字段较多,在客户端里就会暴露过多的细节。而我们希望的是,在不破坏封装性的前提下,捕获一个对象的内部状态,并在该对象之外保存这个状态。这句话的后面两句是啥意思呢?捕获一个对象的内部状态——值得是要获得需要备份的状态(或者简单理解为字段),该备份状态要保存在另外一个类并要有对象自己来读取。

我们还是先来看代码。在备忘录模式中涉及到三个基本的类,一个是原始类,即需要备份的状态类,一个是备份类,即具体存储状态,还有一个管理者,用来提供备份状态类。

首先实现原始类,在这个类里除去该类本身自有的方法,还要有一个创建备份和获取备份的两个方法。

代码语言:javascript复制
 1 package day_27_memento;
 2 
 3 /**
 4  * 发起人,它要负责创建一个备忘录Memento用来记录当前时刻它的状态
 5  * @author turbo
 6  *
 7  * 2016年9月27日
 8  */
 9 public class Originator {
10     private String state;
11 
12     public String getState() {
13         return state;
14     }
15 
16     public void setState(String state) {
17         this.state = state;
18     }
19     
20     /**
21      * 显示状态
22      */
23     public void show(){
24         System.out.println("state : "   state);
25     }
26     
27     /**
28      * 创建备忘录
29      * @return 备份的状态
30      */
31     public Memento createMemento(){
32         return (new Memento(state));  //在这句我们可以看到将状态细节封装在了内部,对外部透明。
33     }
34     
35     /**
36      * 获取备份
37      * @param memento 备份类
38      */
39     public void setMememto(Memento memento){
40         state = memento.getState();
41     }
42 }

备忘类。

代码语言:javascript复制
 1 package day_27_memento;
 2 
 3 /**
 4  * 备忘类
 5  * @author turbo
 6  *
 7  * 2016年9月27日
 8  */
 9 public class Memento {
10     private String state;
11     
12     public String getState() {
13         return state;
14     }
15 
16     public Memento(String state){
17         this.state = state;
18     }
19 }

接着来看看这个管理者类。

代码语言:javascript复制
 1 package day_27_memento;
 2 
 3 /**
 4  * 管理者
 5  * @author turbo
 6  *
 7  * 2016年9月27日
 8  */
 9 public class Caretaker {
10     private Memento memento;
11 
12     public Memento getMemento() {
13         return memento;
14     }
15 
16     public void setMemento(Memento memento) {
17         this.memento = memento;
18     }
19     
20 }

管理者类就是用来提供备忘录类的。最后我们来看客户端代码。

代码语言:javascript复制
 1 package day_27_memento;
 2 
 3 /**
 4  * @author turbo
 5  *
 6  * 2016年9月27日
 7  */
 8 public class Main {
 9     public static void main(String[] args){
10         /*当前状态*/
11         Originator originator = new Originator();
12         originator.setState("On");
13         originator.show();
14         /*备份当前状态*/
15         Caretaker caretaker = new Caretaker();
16         caretaker.setMemento(originator.createMemento());
17         /*更新状态*/
18         originator.setState("Off");
19         originator.show();
20         /*备份状态*/
21         originator.setMememto(caretaker.getMemento());
22         originator.show();
23     }
24 }

从代码16行、21行,我们能看到实际上就是书中所说,“有时一些对象的内部信息必须保存在对象以外的地方,但是必须要由对象自己读取”。

0 人点赞