《设计模式》原型模式

简介: 《设计模式》原型模式

《设计模式》原型模式

定义

原型模式就是指用原型实例指定创建对象的种类,并且通过拷贝这些原型,创建新的对象。

它是一种创建型设计模式,允许一个对象再创建另外一个可定制的对象,无需知道创建的细节。

它的工作原理就是:通过将一个原型对象传给那个要发动创建的对象,这个要发动创建的对象通过请求原型对象拷贝它们自己来实施创建,即对象.clone().

原型模式的 UML 类图如下所示:2e718f6de6d94987ae2f2182c528d9c8.png

其中:

  • Prototype 表示原型类,声明一个克隆自己的接口
  • ConcretePrototype 表示具体的原型类,实现一个克隆自己的操作
  • Client 让一个原型对象克隆自己,从而创建一个新的对象(属性一样)

原型模式的注意事项

创建新的对象比较复杂时,可以使用原型模式简化对象的创建过程,提高效率。

不用重新初始化对象,而是动态地获得对象运行时的状态。

如果原始对象属性发生变化,其他克隆对象也会发生变化,而无需修改代码。

原型模式默认使用浅拷贝克隆对象,如果想实现深拷贝,推荐使用对象序列化方式。

现在有一只羊,姓名为 tom,年龄为1,颜色为白色,需要编写程序创建 10 只和 tom 羊属性完全相同的羊。

使用传统方式对问题进行分析,其 UML 图如下所示:2e718f6de6d94987ae2f2182c528d9c8.png

Sheep

public class Sheep {
    private String name;
    private int age;
    private String color;
    public String getName() {
        return name;
    }
    public void setName(String name) {
        this.name = name;
    }
    public int getAge() {
        return age;
    }
    public void setAge(int age) {
        this.age = age;
    }
    public String getColor() {
        return color;
    }
    public void setColor(String color) {
        this.color = color;
    }
    public Sheep(String name, int age, String color) {
        this.name = name;
        this.age = age;
        this.color = color;
    }
}

Client

public class Client {
    public static void main(String[] args) {
        Sheep sheep = new Sheep("tom", 1, "白色");
        Sheep sheep1 = new Sheep(sheep.getName(), sheep.getAge(), sheep.getColor());
        Sheep sheep2 = new Sheep(sheep.getName(), sheep.getAge(), sheep.getColor());
        Sheep sheep3 = new Sheep(sheep.getName(), sheep.getAge(), sheep.getColor());
        //...
    }
}

传统实现方式的优点是简单易操作,也很容易理解。但是,在创建新的对象时,总是需要重新获取原始对象的属性,如果创建的对象比较复杂时,效率较为低下。此外,总是需要重新初始化对象,而不是动态地获得对象运行时的状态,不够灵活。


使用原型模式解决这个问题,需要让 Sheep 类实现 Cloneable 接口中的 clone 方法,这样就可以让 Sheep 类具有复制的能力,让程序具有更高的效率和扩展性。

Sheep

public class Sheep implements Cloneable{
    private String name;
    private int age;
    private String color;
    public Sheep friend;  //默认是浅拷贝,直接拷贝地址
    private String address = "山羊";
    @Override
    public Object clone() {
        Sheep sheep = null;
        try {
            sheep = (Sheep) super.clone(); //默认是浅拷贝,直接拷贝地址
        } catch (Exception e) {
            e.printStackTrace();
        }
        return sheep;
    }
    @Override
    public String toString() {
        return "Sheep{" +
                "name='" + name + '\'' +
                ", age=" + age +
                ", color='" + color + '\'' +
                ", address='" + address + '\'' +
                '}';
    }
    public String getName() {
        return name;
    }
    public void setName(String name) {
        this.name = name;
    }
    public int getAge() {
        return age;
    }
    public void setAge(int age) {
        this.age = age;
    }
    public String getColor() {
        return color;
    }
    public void setColor(String color) {
        this.color = color;
    }
    public Sheep(String name, int age, String color) {
        this.name = name;
        this.age = age;
        this.color = color;
    }
}

Client

public class Client {
    public static void main(String[] args) {
        Sheep sheep = new Sheep("tom", 1, "白色");
        sheep.friend = new Sheep("jack", 2, "黑色");
        Sheep sheep1 = (Sheep) sheep.clone(); //sheep1 != sheep,但是 sheep.friend==sheep1.friend
        Sheep sheep2 = (Sheep) sheep.clone(); //sheep2 != sheep,但是 sheep.friend==sheep2.friend
        Sheep sheep3 = (Sheep) sheep.clone(); 
    }
}

以上就是使用原型模式解决克隆羊问题的具体实现。

原型模式在 Spring 框架中也有应用, AbstractBeanFactory 类中的 doGetBean 方法部分代码如下:

protected <T> T doGetBean(String name, @Nullable Class<T> requiredType, @Nullable Object[] args, boolean typeCheckOnly) throws BeansException {
  if (mbd.isSingleton()) {
         sharedInstance = this.getSingleton(beanName, () -> {
              try {
                  return this.createBean(beanName, mbd, args);
              } catch (BeansException var5) {
                  this.destroySingleton(beanName);
                  throw var5;
              }
          });
          bean = this.getObjectForBeanInstance(sharedInstance, name, beanName, mbd);
      } else if (mbd.isPrototype()) {
          var11 = null;
          Object prototypeInstance;
          try {
              this.beforePrototypeCreation(beanName);
              prototypeInstance = this.createBean(beanName, mbd, args);
          } finally {
              this.afterPrototypeCreation(beanName);
          }
          bean = this.getObjectForBeanInstance(prototypeInstance, name, beanName, mbd);
      } else {
        //...
      }
      //...
}
相关文章
|
6月前
|
设计模式 安全 Java
面向对象编程的精髓:Java设计模式 - 原型模式(Prototype)完全参考手册
【4月更文挑战第7天】原型模式是OOP中的创建型设计模式,用于通过复制现有实例创建新实例,尤其适用于创建成本高或依赖其他对象的情况。它包括Prototype接口、ConcretePrototype实现和Client客户端角色。优点是性能优化、避免子类化和动态增加产品族。实现包括定义原型接口、实现具体原型和客户端调用克隆方法。最佳实践涉及确保克隆正确性、选择深拷贝或浅拷贝及考虑线程安全。但需注意克隆方法管理、性能开销和循环引用等问题。在Java中,实现Cloneable接口和覆盖clone方法可实现原型模式。
77 4
|
6月前
|
设计模式 Java 关系型数据库
23种设计模式 —— 原型模式【克隆羊、浅拷贝、深拷贝】
23种设计模式 —— 原型模式【克隆羊、浅拷贝、深拷贝】
|
6月前
|
设计模式 安全 Java
【设计模式】原型模式
【设计模式】原型模式
|
2月前
|
设计模式 Java 关系型数据库
【Java笔记+踩坑】设计模式——原型模式
对比原型模式和传统方式的实现思路、代码方案、优缺点,阐述原型模式的使用场景,以及深拷贝、浅拷贝等相关概念,并扩展原型模式在Spring源码中的应用。
【Java笔记+踩坑】设计模式——原型模式
|
2月前
|
设计模式 Java
Java设计模式-原型模式(3)
Java设计模式-原型模式(3)
Java设计模式-原型模式(3)
|
4月前
|
设计模式
iLogtail设计模式问题之iLogtail中的原型模式是什么
iLogtail设计模式问题之iLogtail中的原型模式是什么
iLogtail设计模式问题之iLogtail中的原型模式是什么
|
4月前
|
设计模式 JavaScript
js设计模式【详解】—— 原型模式
js设计模式【详解】—— 原型模式
51 6
|
5月前
|
设计模式 Java
Java设计模式之原型模式详解
Java设计模式之原型模式详解
|
5月前
|
设计模式
原型模式-大话设计模式
原型模式-大话设计模式
|
5月前
|
设计模式 Java Spring
设计模式——原型模式
设计模式——原型模式

热门文章

最新文章

  • 1
    C++一分钟之-设计模式:工厂模式与抽象工厂
    43
  • 2
    《手把手教你》系列基础篇(九十四)-java+ selenium自动化测试-框架设计基础-POM设计模式实现-下篇(详解教程)
    50
  • 3
    C++一分钟之-C++中的设计模式:单例模式
    58
  • 4
    《手把手教你》系列基础篇(九十三)-java+ selenium自动化测试-框架设计基础-POM设计模式实现-上篇(详解教程)
    38
  • 5
    《手把手教你》系列基础篇(九十二)-java+ selenium自动化测试-框架设计基础-POM设计模式简介(详解教程)
    64
  • 6
    Java面试题:结合设计模式与并发工具包实现高效缓存;多线程与内存管理优化实践;并发框架与设计模式在复杂系统中的应用
    59
  • 7
    Java面试题:设计模式在并发编程中的创新应用,Java内存管理与多线程工具类的综合应用,Java并发工具包与并发框架的创新应用
    42
  • 8
    Java面试题:如何使用设计模式优化多线程环境下的资源管理?Java内存模型与并发工具类的协同工作,描述ForkJoinPool的工作机制,并解释其在并行计算中的优势。如何根据任务特性调整线程池参数
    50
  • 9
    Java面试题:请列举三种常用的设计模式,并分别给出在Java中的应用场景?请分析Java内存管理中的主要问题,并提出相应的优化策略?请简述Java多线程编程中的常见问题,并给出解决方案
    112
  • 10
    Java面试题:设计模式如单例模式、工厂模式、观察者模式等在多线程环境下线程安全问题,Java内存模型定义了线程如何与内存交互,包括原子性、可见性、有序性,并发框架提供了更高层次的并发任务处理能力
    78