在不破坏封装性的前提下,捕获一个对象的内部状态,并在该对象之外保存这个状态。这样以后就可以将该对象恢复到原先保存的状态。
适用性
- 必须保存一个对象在某一个时刻的(部分)状态,这样以后需要时它才能恢复到它他先前的状态
- 如果一个用接口来让其他对象直接得到这些状态,将会暴露对象的实现细节并破坏对象的封装性。
参与者
Memento
——备忘录存储原发器对象的内部状态。原发器根据需求决定备忘录存储原发器的那些内部状态
——防止原发器以外的其他对象访问备忘录,备忘录实际上有两个接口,管理者只能看到备忘录的窄接口——它只能将备忘录传递给其他对象。相反,原发器只能看到一个宽接口,允许他访问返回到先前状态所需的所有数据。理想的情况时只允许生成本备忘录的原发器访问本备忘录的内部状态。
Originator
——原发器创建一个备忘录,用已记录当前时刻他的内部状态
——使用备忘录回复内部状态
Caretaker
——负责保存好备忘录
——不能对备忘录的内容进行操作或检查
下面以备份手机通讯录为例子来实现了备忘录模式
1 // 联系人 2 public class ContactPerson 3 { 4 public string Name { get; set; } 5 public string MobileNum { get; set; } 6 }
1 // 发起人 2 public class MobileOwner 3 { 4 // 发起人需要保存的内部状态 5 public List<ContactPerson> ContactPersons { get; set; } 6 7 public MobileOwner(List<ContactPerson> persons) 8 { 9 ContactPersons = persons; 10 } 11 12 // 创建备忘录,将当期要保存的联系人列表导入到备忘录中 13 public ContactMemento CreateMemento() 14 { 15 // 这里也应该传递深拷贝,new List方式传递的是浅拷贝, 16 // 因为ContactPerson类中都是string类型,所以这里new list方式对ContactPerson对象执行了深拷贝 17 // 如果ContactPerson包括非string的引用类型就会有问题,所以这里也应该用序列化传递深拷贝 18 return new ContactMemento(new List<ContactPerson>(this.ContactPersons)); 19 } 20 21 // 将备忘录中的数据备份导入到联系人列表中 22 public void RestoreMemento(ContactMemento memento) 23 { 24 // 下面这种方式是错误的,因为这样传递的是引用, 25 // 则删除一次可以恢复,但恢复之后再删除的话就恢复不了. 26 // 所以应该传递contactPersonBack的深拷贝,深拷贝可以使用序列化来完成 27 this.ContactPersons = memento.contactPersonBack; 28 } 29 30 public void Show() 31 { 32 Console.WriteLine("联系人列表中有{0}个人,他们是:", ContactPersons.Count); 33 foreach (ContactPerson p in ContactPersons) 34 { 35 Console.WriteLine("姓名: {0} 号码为: {1}", p.Name, p.MobileNum); 36 } 37 } 38 }
1 // 备忘录 2 public class ContactMemento 3 { 4 // 保存发起人的内部状态 5 public List<ContactPerson> contactPersonBack; 6 7 public ContactMemento(List<ContactPerson> persons) 8 { 9 contactPersonBack = persons; 10 } 11 }
1 // 管理角色 2 public class Caretaker 3 { 4 public ContactMemento ContactM { get; set; } 5 }
1 class Program 2 { 3 static void Main(string[] args) 4 { 5 List<ContactPerson> persons = new List<ContactPerson>() 6 { 7 new ContactPerson() { Name= "Learning Hard", MobileNum = "123445"}, 8 new ContactPerson() { Name = "Tony", MobileNum = "234565"}, 9 new ContactPerson() { Name = "Jock", MobileNum = "231455"} 10 }; 11 MobileOwner mobileOwner = new MobileOwner(persons); 12 mobileOwner.Show(); 13 14 // 创建备忘录并保存备忘录对象 15 Caretaker caretaker = new Caretaker(); 16 caretaker.ContactM = mobileOwner.CreateMemento(); 17 18 // 更改发起人联系人列表 19 Console.WriteLine("----移除最后一个联系人--------"); 20 mobileOwner.ContactPersons.RemoveAt(2); 21 mobileOwner.Show(); 22 23 // 恢复到原始状态 24 Console.WriteLine("-------恢复联系人列表------"); 25 mobileOwner.RestoreMemento(caretaker.ContactM); 26 mobileOwner.Show(); 27 28 Console.Read(); 29 } 30 }
刚开始通讯录中有3个联系人,然后移除以后一个后变成2个联系人了,最后恢复原来的联系人列表后,联系人列表中又恢复为3个联系人了。
上面代码只是保存了一个还原点,即备忘录中只保存了3个联系人的数据,但是,如果想备份多个还原点怎么办呢?即恢复到3个人后,又想恢复到前面2个人的状态,这时候可能你会想,这样没必要啊,到时候在删除不就好了。但是如果在实际应用中,可能我们发了很多时间去创建通讯录中只有2个联系人的状态,恢复到3个人的状态后,发现这个状态时错误的,还是原来2个人的状态是正确的,难道我们又去花之前的那么多时间去重复操作吗?这显然不合理,如果就思考,能不能保存多个还原点呢?保存多个还原点其实很简单,只需要保存多个备忘录对象就可以了。具体实现代码如下所示:
1 // 联系人 2 public class ContactPerson 3 { 4 public string Name { get; set; } 5 public string MobileNum { get; set; } 6 }
1 // 发起人 2 public class MobileOwner 3 { 4 public List<ContactPerson> ContactPersons { get; set; } 5 public MobileOwner(List<ContactPerson> persons) 6 { 7 ContactPersons = persons; 8 } 9 10 // 创建备忘录,将当期要保存的联系人列表导入到备忘录中 11 public ContactMemento CreateMemento() 12 { 13 // 这里也应该传递深拷贝,new List方式传递的是浅拷贝, 14 // 因为ContactPerson类中都是string类型,所以这里new list方式对ContactPerson对象执行了深拷贝 15 // 如果ContactPerson包括非string的引用类型就会有问题,所以这里也应该用序列化传递深拷贝 16 return new ContactMemento(new List<ContactPerson>(this.ContactPersons)); 17 } 18 19 // 将备忘录中的数据备份导入到联系人列表中 20 public void RestoreMemento(ContactMemento memento) 21 { 22 if (memento != null) 23 { 24 // 下面这种方式是错误的,因为这样传递的是引用, 25 // 则删除一次可以恢复,但恢复之后再删除的话就恢复不了. 26 // 所以应该传递contactPersonBack的深拷贝,深拷贝可以使用序列化来完成 27 this.ContactPersons = memento.ContactPersonBack; 28 } 29 } 30 public void Show() 31 { 32 Console.WriteLine("联系人列表中有{0}个人,他们是:", ContactPersons.Count); 33 foreach (ContactPerson p in ContactPersons) 34 { 35 Console.WriteLine("姓名: {0} 号码为: {1}", p.Name, p.MobileNum); 36 } 37 } 38 }
1 // 备忘录 2 public class ContactMemento 3 { 4 public List<ContactPerson> ContactPersonBack {get;set;} 5 public ContactMemento(List<ContactPerson> persons) 6 { 7 ContactPersonBack = persons; 8 } 9 }
1 // 管理角色 2 public class Caretaker 3 { 4 // 使用多个备忘录来存储多个备份点 5 public Dictionary<string, ContactMemento> ContactMementoDic { get; set; } 6 public Caretaker() 7 { 8 ContactMementoDic = new Dictionary<string, ContactMemento>(); 9 } 10 }
1 class Program 2 { 3 static void Main(string[] args) 4 { 5 List<ContactPerson> persons = new List<ContactPerson>() 6 { 7 new ContactPerson() { Name= "Learning Hard", MobileNum = "123445"}, 8 new ContactPerson() { Name = "Tony", MobileNum = "234565"}, 9 new ContactPerson() { Name = "Jock", MobileNum = "231455"} 10 }; 11 12 MobileOwner mobileOwner = new MobileOwner(persons); 13 mobileOwner.Show(); 14 15 // 创建备忘录并保存备忘录对象 16 Caretaker caretaker = new Caretaker(); 17 caretaker.ContactMementoDic.Add(DateTime.Now.ToString(), mobileOwner.CreateMemento()); 18 19 // 更改发起人联系人列表 20 Console.WriteLine("----移除最后一个联系人--------"); 21 mobileOwner.ContactPersons.RemoveAt(2); 22 mobileOwner.Show(); 23 24 // 创建第二个备份 25 Thread.Sleep(1000); 26 caretaker.ContactMementoDic.Add(DateTime.Now.ToString(), mobileOwner.CreateMemento()); 27 28 // 恢复到原始状态 29 Console.WriteLine("-------恢复联系人列表,请从以下列表选择恢复的日期------"); 30 var keyCollection = caretaker.ContactMementoDic.Keys; 31 foreach (string k in keyCollection) 32 { 33 Console.WriteLine("Key = {0}", k); 34 } 35 while (true) 36 { 37 Console.Write("请输入数字,按窗口的关闭键退出:"); 38 39 int index = -1; 40 try 41 { 42 index = Int32.Parse(Console.ReadLine()); 43 } 44 catch 45 { 46 Console.WriteLine("输入的格式错误"); 47 continue; 48 } 49 50 ContactMemento contactMentor = null; 51 if (index < keyCollection.Count && caretaker.ContactMementoDic.TryGetValue(keyCollection.ElementAt(index), out contactMentor)) 52 { 53 mobileOwner.RestoreMemento(contactMentor); 54 mobileOwner.Show(); 55 } 56 else 57 { 58 Console.WriteLine("输入的索引大于集合长度!"); 59 } 60 } 61 } 62 } 63 }
备忘录模式主要思想是——利用备忘录对象来对保存发起人的内部状态,当发起人需要恢复原来状态时,再从备忘录对象中进行获取,在实际开发过程也应用到这点,例如数据库中的事务处理。