原型模式

妖狐艹你老母 2021-08-31 02:55 527阅读 0赞

原型模式

    1. 简介
    1. 原型模式介绍
    • 2.1 定义
    • 2.2 适用场景
    • 2.3 模式分析
    • 2.4 原型模式优点
    • 2.4 原型模式缺点
    1. 示例程序
    • 3.1 Product接口 (Prototype)
    • 3.2 Manager类(Client)
    • 3.3 MessageBox类(ConcreteProtorype)
    • 3.4 UnderlinePen类(ConcreteProtorype)
    • 3.5 Main类
  • 4 原型模式的角色分析
    • 4.1 Prototype(抽象原型类)
    • 4.2 ConcretePrototype(具体原型类)
    • 4.3 Client(客户类/使用者)
    • 4.4 Prototype模式的类图
    1. 原型模式的实际应用案例

1. 简介

  1. 在面向对象系统中,使用原型模式来复制一个对象自身,从而克隆出多个与原型对象一模一样的对象。
  2. 另外在软件系统中,有些对象的创建过程较为复杂,而且有时候需要频繁创建,原型模式通过给出一个原型对象来指明所要创建的对象的类型,然后用复制这个原型对象的办法创建出更多同类型的对象,这就是原型模式的意图所在。

在这里插入图片描述

2. 原型模式介绍

2.1 定义

  1. 使用原型实例指定将要创建的对象类型,通过复制这个实例创建新的对象。

2.2 适用场景

  1. 我们通常都是使用`new`关键字指定类名生成类的实例。
  2. 使用`new`关键字创建类的时候必须指定类名,但是在开发过程中也会有“在不指定类名的前提下生成实例”的需求。例如,在下面这些情况下,就需要根据现有的实例来生成新的实例。
  1. 对象种类繁多,无法将他们整合到一个类的时候;
  2. 难以根据类生成实例时;
  3. 想解耦框架与生成的实例时。

    如果想要让生成实例的框架不再依赖于具体的类,这时,不能指定类名来生成实例,而要事先“注册”一个“原型”实例,然后通过复制该实例来生成新的实例。

2.3 模式分析

  1. 在原型模式结构中定义了一个抽象原型类,所有的`Java`类都继承自`java.lang.Object`,而`Object`类提供一个`clone()`方法,可以将一个`Java`对象复制一份。因此在`Java`中可以直接使用`Object`提供的`clone()`方法来实现对象的克隆,`Java`语言中的原型模式实现很简单。
  2. 能够实现克隆的`Java`类必须实现一个标识接口`Cloneable`,表示这个`Java`类支持复制。如果一个类没有实现这个接口但是调用了`clone()`方法,Java编译器将抛出一个`CloneNotSupportedException`异常。
  3. 注意: `java.lang.Cloneable` 只是起到告诉程序可以调用`clone`方法的作用,它本身并没有定义任何方法。
  4. 在使用原型模式克隆对象时,根据其成员对象是否也克隆,原型模式可以分为两种形式:深克隆 浅克隆
  5. [详解 Java 中的 clone() 方法][Java _ clone_]

2.4 原型模式优点

  • 当创建新的对象实例较为复杂时,使用原型模式可以简化对象的创建过程,通过一个已有实例可以提高新实例的创建效率。
  • 可以动态增加或减少产品类。
  • 原型模式提供了简化的创建结构。
  • 可以使用深克隆的方式保存对象的状态。

2.4 原型模式缺点

  • 需要为每一个类配备一个克隆方法,而且这个克隆方法需要对类的功能进行通盘考虑,这对全新的类来说不是很难,但对已有的类进行改造时,不一定是件容易的事,必须修改其源代码,违背了“开闭原则”。
  • 在实现深克隆时需要编写较为复杂的代码。

3. 示例程序

  1. 下面示例程序的作用是将字符串放入方框中显示出来或者是加了下划线显示出来。
  2. 类和接口一览表:

在这里插入图片描述
示例程序类图:

在这里插入图片描述

3.1 Product接口 (Prototype)

  1. `Product`接口是复制功能接口,该接口继承了`java.lang.Cloneable`(只有实现了该接口的类的实例才可以调用`clone()`方法复制实例,否则会抛出异常).
  2. 另外需要注意:`java.lang.Cloneable` 只是起到告诉程序可以调用`clone`方法的作用,它本身并没有定义任何方法。
  3. public interface Product extends Cloneable{
  4. // use 方法是用于“使用”的方法,具体怎么“使用”,则被交给子类去实现。
  5. public abstract void use(String s);
  6. // creatClone 方法是用于复制实例的方法
  7. public abstract Product creatClone();
  8. }

3.2 Manager类(Client)

  1. `Manager`类使用`Product`接口来复制实例。
  2. `Product`接口以及`Manager`类的代码完全没有出现在`MessageBox`类和`UnderlinePen`类的名字,因此这意味着我们可以独立地修改`Product`接口以及`Manage`r类,不受`MessageBox`类和`UnderlinePen`类的影响。这是非常重要的,因为 一旦在类中使用到了别的类名,就意味着该类与其他类紧密的地耦合在了一起 。在`Manager`类中,并没有写明具体的类名, 仅仅使用了`Product`这个接口名。也就是说,`Product`接口成为了连接`Manager`类与其他具体类之间的桥梁。
  3. import java.util.HashMap;
  4. public class Manager {
  5. // 保存实例的“名字”和“实例”之间的对应关系
  6. private HashMap<String, Product> showcase=new HashMap<String, Product>();
  7. // register 方法将接收到的一组“名字”和“Product接口”注册到showcase中。
  8. // 这里Product是实现Product接口的实例,具体还未确定
  9. public void register(String name ,Product product){
  10. showcase.put(name, product);
  11. }
  12. public Product create(String productname){
  13. Product p=showcase.get(productname);
  14. return p.creatClone();
  15. }
  16. }

3.3 MessageBox类(ConcreteProtorype)

  1. 装饰方框样式的具体原型,实现了`Product`接口,实现复制现有实例并生成新实例的方法。
  2. public class MessageBox implements Product {
  3. //保存的是装饰方框使用的字符样式
  4. private char decochar;
  5. public MessageBox(char decochar) {
  6. this.decochar = decochar;
  7. }
  8. @Override
  9. public void use(String s) {
  10. int length=s.getBytes().length;
  11. for (int i = 0; i < length+4; i++) {
  12. System.out.print(decochar);
  13. }
  14. System.out.println("");
  15. System.out.println(decochar+" "+s+" "+decochar);
  16. for (int i = 0; i < length+4; i++) {
  17. System.out.print(decochar);
  18. }
  19. System.out.println("");
  20. }
  21. //该方法用于复制自己
  22. @Override
  23. public Product creatClone() {
  24. Product p=null;
  25. try {
  26. p=(Product) clone();
  27. } catch (CloneNotSupportedException e) {
  28. e.printStackTrace();
  29. }
  30. return p;
  31. }
  32. }
  33. 只有类自己(或是它的子类)能够调用`Java`语言中定义的`clone`方法。当其他类要求复制实例时,必须先调用`createClone`这样的方法,然后在该方法内部在调用`clone`方法。

3.4 UnderlinePen类(ConcreteProtorype)

  1. 下划线样式的具体原型,实现了`Product`接口,用于实现复制现有实例并生成新实例的方法。`UnderlinePen`类的实现几乎和`MessageBox`类一样,不同的可能只是`use`方法的实现。
  2. public class UnderlinePen implements Product {
  3. private char ulchar;
  4. public UnderlinePen(char ulchar) {
  5. this.ulchar = ulchar;
  6. }
  7. @Override
  8. public void use(String s) {
  9. int length = s.getBytes().length;
  10. System.out.println("\""+s+"\"");
  11. for (int i = 0; i <length+2; i++) {
  12. System.out.print(ulchar);
  13. }
  14. System.out.println("");
  15. }
  16. @Override
  17. public Product creatClone() {
  18. Product p=null;
  19. try {
  20. p=(Product) clone();
  21. } catch (CloneNotSupportedException e) {
  22. e.printStackTrace();
  23. }
  24. return p;
  25. }
  26. }

3.5 Main类

  1. `Main`类首先生成`Manager`实例。接着,在`Manager`实例中通过`register`方法注册了`UnderlinePen`类的实例(带名字)和`MessageBox`类的实例(带名字)。
  2. public class Main {
  3. public static void main(String[] args) {
  4. Manager manager = new Manager();
  5. UnderlinePen underlinePen=new UnderlinePen('~');
  6. MessageBox mbox=new MessageBox('*');
  7. MessageBox sbox=new MessageBox('/');
  8. manager.register("Strong message", underlinePen);
  9. manager.register("Waring Box", mbox);
  10. manager.register("Slash Box", sbox);
  11. Product p1=manager.create("Strong message");
  12. p1.use("hello world");
  13. Product p2=manager.create("Waring Box");
  14. p2.use("hello world");
  15. Product p3=manager.create("Slash Box");
  16. p3.use("hello world");
  17. }
  18. }

在这里插入图片描述

4 原型模式的角色分析

  1. 通过上面的例子,相信大家对于原型模式有了更进一步的认识,下面我们看看原型模式的几个登场角色。

4.1 Prototype(抽象原型类)

  1. `Product`角色负责定义用于复制现有实例来生成新实例的方法。在示例程序中的`Product`接口就是该角色。

4.2 ConcretePrototype(具体原型类)

  1. `ConcretePrototype`角色负责实现复制现有实例并生成新实例的方法。在示例程序中,`MessageBox``UnderlinePen`都是该角色。

4.3 Client(客户类/使用者)

  1. `Client`角色负责使用复制实例的方法生成新的实例。在示例程序中,`Manager`类扮演的就是该角色。

4.4 Prototype模式的类图

在这里插入图片描述

5. 原型模式的实际应用案例

  1. 原型模式应用于很多软件中,如果每次创建一个对象要花大量时间,原型模式是最好的解决方案。很多软件提供的复制(Ctrl + C)和粘贴(Ctrl + V)操作就是原型模式的应用,复制得到的对象与原型对象是两个类型相同但内存地址不同的对象,通过原型模式可以大大提高对象的创建效率。
  2. Struts2中为了保证线程的安全性,Action对象的创建使用了原型模式,访问一个已经存在的Action对象时将通过克隆的方式创建出一个新的对象,从而保证其中定义的变量无须进行加锁实现同步,每一个Action中都有自己的成员变量,避免Struts1因使用单例模式而导致的并发和同步问题。
  3. Spring中,用户也可以采用原型模式来创建新的bean实例,从而实现每次获取的是通过克隆生成的新实例,对其进行修改时对原有实例对象不造成任何影响。

发表评论

表情:
评论列表 (有 0 条评论,527人围观)

还没有评论,来说两句吧...

相关阅读

    相关 原型模式

    一. 原型模式简介    原型模式(Prototype Pattern)也是一种创建型模式,它关注的是大量相似对象的创建问题。我们经常会遇到这样的情况:在系统中要创建大量的

    相关 原型模式

    对象浅拷贝 实现Clonable接口,重写clone方法,jvm的本地方法实现拷贝 Object protected native Object clone()

    相关 原型模式

    原型模式(Prototype Pattern):通过原型实例指定创建对象的种类,并且通过克隆这些原型来创建新的对象。原型模式是一种创建型模式   完成原型模式一般需要3个角

    相关 原型模式

    原型模式:用原型实例指定创建指定对象的种类,并且通过复制这些原型创建新的对象,同时又能保证性能。这种模式是实现了一个原型接口,该接口用于创建当前对象的克隆。当直接创建对象的代价

    相关 原型模式

    原型模式 定义 在软件系统中,有时需要多次创建某一类型的对象,为了简化创建过程,可以只创建一个对象,然后通过对象克隆的方式复制出多个相同的对象,这就是原型设计模

    相关 原型模式

    型模式(Prototype Pattern)是用于创建重复的对象,同时又能保证性能。这种类型的设计模式属于创建型模式,它提供了一种创建对象的最佳方式。 这种模式是实现了...