对象的克隆——原型模式(四)

7.5 原型管理器的引入和实现

原型管理器(Prototype Manager)是将多个原型对象存储在一个集合中供客户端使用,它是一个专门负责克隆对象的工厂,其中定义了一个集合用于存储原型对象,如果需要某个原型对象的一个克隆,可以通过复制集合中对应的原型对象来获得。在原型管理器中针对抽象原型类进行编程,以便扩展。其结构如图7-8所示:

对象的克隆——原型模式(四) - 图1

图7-8 带原型管理器的原型模式

下面通过模拟一个简单的公文管理器来介绍原型管理器的设计与实现: Sunny软件公司在日常办公中有许多公文需要创建、递交和审批,例如《可行性分析报告》、《立项建议书》、《软件需求规格说明书》、《项目进展报告》等,为了提高工作效率,在OA系统中为各类公文均创建了模板,用户可以通过这些模板快速创建新的公文,这些公文模板需要统一进行管理,系统根据用户请求的不同生成不同的新公文。

我们使用带原型管理器的原型模式实现公文管理器的设计,其结构如图7-9所示:

对象的克隆——原型模式(四) - 图2

图7-9 公文管理器结构图

以下是实现该功能的一些核心代码,考虑到代码的可读性,我们对所有的类都进行了简化:

  1. import java.util.*;
  2. //抽象公文接口,也可定义为抽象类,提供clone()方法的实现,将业务方法声明为抽象方法
  3. interface OfficialDocument extends Cloneable
  4. {
  5. public OfficialDocument clone();
  6. public void display();
  7. }
  8. //可行性分析报告(Feasibility Analysis Report)类
  9. class FAR implements OfficialDocument
  10. {
  11. public OfficialDocument clone()
  12. {
  13. OfficialDocument far = null;
  14. try
  15. {
  16. far = (OfficialDocument)super.clone();
  17. }
  18. catch(CloneNotSupportedException e)
  19. {
  20. System.out.println("不支持复制!");
  21. }
  22. return far;
  23. }
  24. public void display()
  25. {
  26. System.out.println("《可行性分析报告》");
  27. }
  28. }
  29. //软件需求规格说明书(Software Requirements Specification)类
  30. class SRS implements OfficialDocument
  31. {
  32. public OfficialDocument clone()
  33. {
  34. OfficialDocument srs = null;
  35. try
  36. {
  37. srs = (OfficialDocument)super.clone();
  38. }
  39. catch(CloneNotSupportedException e)
  40. {
  41. System.out.println("不支持复制!");
  42. }
  43. return srs;
  44. }
  45. public void display()
  46. {
  47. System.out.println("《软件需求规格说明书》");
  48. }
  49. }
  50. //原型管理器(使用饿汉式单例实现)
  51. class PrototypeManager
  52. {
  53. //定义一个Hashtable,用于存储原型对象
  54. private Hashtable ht=new Hashtable();
  55. private static PrototypeManager pm = new PrototypeManager();
  56. //为Hashtable增加公文对象
  57. private PrototypeManager()
  58. {
  59. ht.put("far",new FAR());
  60. ht.put("srs",new SRS());
  61. }
  62. //增加新的公文对象
  63. public void addOfficialDocument(String key,OfficialDocument doc)
  64. {
  65. ht.put(key,doc);
  66. }
  67. //通过浅克隆获取新的公文对象
  68. public OfficialDocument getOfficialDocument(String key)
  69. {
  70. return ((OfficialDocument)ht.get(key)).clone();
  71. }
  72. public static PrototypeManager getPrototypeManager()
  73. {
  74. return pm;
  75. }
  76. }

客户端代码如下所示:

  1. class Client
  2. {
  3. public static void main(String args[])
  4. {
  5. //获取原型管理器对象
  6. PrototypeManager pm = PrototypeManager.getPrototypeManager();
  7. OfficialDocument doc1,doc2,doc3,doc4;
  8. doc1 = pm.getOfficialDocument("far");
  9. doc1.display();
  10. doc2 = pm.getOfficialDocument("far");
  11. doc2.display();
  12. System.out.println(doc1 == doc2);
  13. doc3 = pm.getOfficialDocument("srs");
  14. doc3.display();
  15. doc4 = pm.getOfficialDocument("srs");
  16. doc4.display();
  17. System.out.println(doc3 == doc4);
  18. }
  19. }

编译并运行程序,输出结果如下:

  1. 《可行性分析报告》
  2. 《可行性分析报告》
  3. false
  4. 《软件需求规格说明书》
  5. 《软件需求规格说明书》
  6. false

在PrototypeManager中定义了一个Hashtable类型的集合对象,使用“键值对”来存储原型对象,客户端可以通过Key(如“far”或“srs”)来获取对应原型对象的克隆对象。PrototypeManager类提供了类似工厂方法的getOfficialDocument()方法用于返回一个克隆对象。在本实例代码中,我们将PrototypeManager设计为单例类,使用饿汉式单例实现,确保系统中有且仅有一个PrototypeManager对象,有利于节省系统资源,并可以更好地对原型管理器对象进行控制。

思考

如果需要增加一种新类型的公文,如《项目进展报告》(Project Progress Report, PPR),公文管理器系统源代码如何修改,动手实践你的修改方案。

7.6 原型模式总结

原型模式作为一种快速创建大量相同或相似对象的方式,在软件开发中应用较为广泛,很多软件提供的复制(Ctrl + C)和粘贴(Ctrl + V)操作就是原型模式的典型应用,下面对该模式的使用效果和适用情况进行简单的总结。

1.主要优点

原型模式的主要优点如下:

(1) 当创建新的对象实例较为复杂时,使用原型模式可以简化对象的创建过程,通过复制一个已有实例可以提高新实例的创建效率。

(2) 扩展性较好,由于在原型模式中提供了抽象原型类,在客户端可以针对抽象原型类进行编程,而将具体原型类写在配置文件中,增加或减少产品类对原有系统都没有任何影响。

(3) 原型模式提供了简化的创建结构,工厂方法模式常常需要有一个与产品类等级结构相同的工厂等级结构,而原型模式就不需要这样,原型模式中产品的复制是通过封装在原型类中的克隆方法实现的,无须专门的工厂类来创建产品。

(4) 可以使用深克隆的方式保存对象的状态,使用原型模式将对象复制一份并将其状态保存起来,以便在需要的时候使用(如恢复到某一历史状态),可辅助实现撤销操作。

2.主要缺点

原型模式的主要缺点如下:

(1) 需要为每一个类配备一个克隆方法,而且该克隆方法位于一个类的内部,当对已有的类进行改造时,需要修改源代码,违背了“开闭原则”。

(2) 在实现深克隆时需要编写较为复杂的代码,而且当对象之间存在多重的嵌套引用时,为了实现深克隆,每一层对象对应的类都必须支持深克隆,实现起来可能会比较麻烦。

3.适用场景 在以下情况下可以考虑使用原型模式:

(1) 创建新对象成本较大(如初始化需要占用较长的时间,占用太多的CPU资源或网络资源),新的对象可以通过原型模式对已有对象进行复制来获得,如果是相似对象,则可以对其成员变量稍作修改。

(2) 如果系统要保存对象的状态,而对象的状态变化很小,或者对象本身占用内存较少时,可以使用原型模式配合备忘录模式来实现。

(3) 需要避免使用分层次的工厂类来创建分层次的对象,并且类的实例对象只有一个或很少的几个组合状态,通过复制原型对象得到新实例可能比使用构造函数创建一个新实例更加方便。

练习

设计并实现一个客户类Customer,其中包含一个名为客户地址的成员变量,客户地址的类型为Address,用浅克隆和深克隆分别实现Customer对象的复制并比较这两种克隆方式的异同。