備忘錄這個詞彙你們應該都不陌生,我就常用備忘錄來記錄一些比較重要的或者容易遺忘的信息,與之相關的最多見的應用有許多,好比遊戲存檔,咱們玩遊戲的時候確定有存檔功能,旨在下一次登陸游戲時能夠從上次退出的地方繼續遊戲,或者對復活點進行存檔,若是掛掉了則能夠讀取復活點的存檔信息從新開始。與之相相似的就是數據庫的事務回滾,或者重作日誌redo log等。git
備忘錄模式(Memento),在不破壞封裝性的前提下,捕獲一個對象的內部狀態,並在該對象以外保存着這個狀態。這樣之後就可將該對象恢復到原先保存的狀態。UML結構圖以下:數據庫
其中,Originator是發起人,負責建立一個備忘錄Memento,用以記錄當前時刻它的內部狀態,並可以使用備忘錄恢復內部狀態;Memento是備忘錄,負責存儲Originator對象的內部狀態,並可防止Originator之外的其餘對象訪問備忘錄Memento;Caretaker是管理者,負責保存好備忘錄的Memento,不能對備忘錄的內容進行操做或檢查。瀏覽器
記錄當前時刻的內部狀態,並負責建立和恢復備忘錄數據,容許訪問返回到先前狀態所需的全部數據。測試
1 public class Originator { 2 3 private String state; 4 5 public String getState() { 6 return state; 7 } 8 9 public void setState(String state) { 10 this.state = state; 11 } 12 13 public Memento createMento() { 14 return (new Memento(state)); 15 } 16 17 public void setMemento(Memento memento) { 18 state = memento.getState(); 19 } 20 21 public void show() { 22 System.out.println("state = " + state); 23 } 24 25 }
負責存儲Originator發起人對象的內部狀態,在須要的時候提供發起人須要的內部狀態。this
1 public class Memento { 2 3 private String state; 4 5 public Memento(String state) { 6 this.state = state; 7 } 8 9 public String getState() { 10 return state; 11 } 12 13 }
對備忘錄進行管理、保存和提供備忘錄,只能將備忘錄傳遞給其餘角色。spa
1 public class Caretaker { 2 3 private Memento memento; 4 5 public Memento getMemento() { 6 return memento; 7 } 8 9 public void setMemento(Memento memento) { 10 this.memento = memento; 11 } 12 13 }
下面編寫一小段代碼測試一下,即先將狀態置爲On,保存後再將狀態置爲Off,而後經過備忘錄管理員角色恢復初始狀態。日誌
1 public class Client { 2 3 public static void main(String[] args) { 4 Originator originator = new Originator(); 5 originator.setState("On"); //Originator初始狀態 6 originator.show(); 7 8 Caretaker caretaker = new Caretaker(); 9 caretaker.setMemento(originator.createMento()); 10 11 originator.setState("Off"); //Originator狀態變爲Off 12 originator.show(); 13 14 originator.setMemento(caretaker.getMemento()); //回覆初始狀態 15 originator.show(); 16 } 17 18 }
運行結果以下:code
下面以遊戲存檔爲例,看一下如何用備忘錄模式實現。UML圖以下:對象
簡單記錄了遊戲角色的生命力、攻擊力、防護力,經過saveState()方法來保存當前狀態,經過recoveryState()方法來恢復角色狀態。blog
1 public class GameRole { 2 3 private int vit; //生命力 4 private int atk; //攻擊力 5 private int def; //防護力 6 7 public int getVit() { 8 return vit; 9 } 10 public void setVit(int vit) { 11 this.vit = vit; 12 } 13 public int getAtk() { 14 return atk; 15 } 16 public void setAtk(int atk) { 17 this.atk = atk; 18 } 19 public int getDef() { 20 return def; 21 } 22 public void setDef(int def) { 23 this.def = def; 24 } 25 26 //狀態顯示 27 public void stateDisplay() { 28 System.out.println("角色當前狀態:"); 29 System.out.println("體力:" + this.vit); 30 System.out.println("攻擊力:" + this.atk); 31 System.out.println("防護力: " + this.def); 32 System.out.println("-----------------"); 33 } 34 35 //得到初始狀態 36 public void getInitState() { 37 this.vit = 100; 38 this.atk = 100; 39 this.def = 100; 40 } 41 42 //戰鬥後 43 public void fight() { 44 this.vit = 0; 45 this.atk = 0; 46 this.def = 0; 47 } 48 49 //保存角色狀態 50 public RoleStateMemento saveState() { 51 return (new RoleStateMemento(vit, atk, def)); 52 } 53 54 //恢復角色狀態 55 public void recoveryState(RoleStateMemento memento) { 56 this.vit = memento.getVit(); 57 this.atk = memento.getAtk(); 58 this.def = memento.getDef(); 59 } 60 61 }
備忘錄類,用於存儲角色狀態。
1 public class RoleStateMemento { 2 3 private int vit; //生命力 4 private int atk; //攻擊力 5 private int def; //防護力 6 7 public RoleStateMemento(int vit, int atk, int def) { 8 this.vit = vit; 9 this.atk = atk; 10 this.def = def; 11 } 12 13 public int getVit() { 14 return vit; 15 } 16 17 public void setVit(int vit) { 18 this.vit = vit; 19 } 20 21 public int getAtk() { 22 return atk; 23 } 24 25 public void setAtk(int atk) { 26 this.atk = atk; 27 } 28 29 public int getDef() { 30 return def; 31 } 32 33 public void setDef(int def) { 34 this.def = def; 35 } 36 37 }
備忘錄管理者。
1 public class RoleStateCaretaker { 2 3 private RoleStateMemento memento; 4 5 public RoleStateMemento getMemento() { 6 return memento; 7 } 8 9 public void setMemento(RoleStateMemento memento) { 10 this.memento = memento; 11 } 12 13 }
下面編寫一個簡單的程序測試一下,編寫邏輯大體爲打boss前存檔,打boss失敗了,讀檔。
1 public class Client { 2 3 public static void main(String[] args) { 4 //打boss前 5 GameRole gameRole = new GameRole(); 6 gameRole.getInitState(); 7 gameRole.stateDisplay(); 8 9 //保存進度 10 RoleStateCaretaker caretaker = new RoleStateCaretaker(); 11 caretaker.setMemento(gameRole.saveState()); 12 13 //打boss失敗 14 gameRole.fight(); 15 gameRole.stateDisplay(); 16 17 //恢復狀態 18 gameRole.recoveryState(caretaker.getMemento()); 19 gameRole.stateDisplay(); 20 } 21 22 }
運行結果以下: