【HeadFirst 设计模式学习笔记】18 原型(Prototype)模式拾零

简介:

1.概述

这个模式在Head First这本书中也没有具体提到。其主要应用于如下场景:对于创建一个新的对象,若且你手中已经有了一个非常相像的对象实例,那么你可以利用这个现有对象进行新对象的创建。用现实中的例子来说,比如细胞分裂,一个细胞分裂为两个细胞(可能这个比喻在生物学上有些出入)。参与这个模式的两种实体为:

原型:一个提供克隆自己的接口的抽象。

具体原型:实现这个克隆自己的的接口的具体实现类。

2.实例

首先我们实现一个原型模式的原始例子,这里利用了Java中Object的clone()方法:

  1: public interface Prototype {
  2:     public abstract Object clone ( );
  3: }
  4: 
  5:  
  6: 
  7: public class ConcretePrototype implements Prototype {
  8:     public Object clone() {
  9:         return super.clone();
 10:     }
 11: }
 12: 
 13: public class Client {
 14: 
 15:     public static void main( String arg[] ) 
 16:     {
 17:         ConcretePrototype obj1= new ConcretePrototype ();
 18:         ConcretePrototype obj2 = ConcretePrototype)obj1.clone();
 19:     }
 20: 
 21: }
 22:  
 23: 

显然上边的这个例子并没有多少实用性,下边的例子中我们要创建一个具有克隆自己功能的表示相关产品的对象,实现上利用了简单工厂方法——这个经常和原型模式一起使用。

我们利用Java中的Cloneable接口定义原型,其中就提供了一个克隆自己的方法的接口定义。然后我们实现这个具体原型,当然相对于标准的原型模式,这里面有增加了一层产品的抽象,因为毕竟Java中的Cloneable是一个接口,需要有一层实现这个接口中clone方法的抽象作为抽象产品类。

  1: public abstract class Product implements Cloneable {
  2:     private String title;
  3:     private String price;
  4:     public String getTitle() {
  5:         return title;
  6:     }
  7:  
  8:     public void setTitle(String title) {
  9:         this.title = title;
 10:     }
 11:  
 12:     public String getPrice() {
 13:         return price;
 14:     }
 15:  
 16:     public void setPrice(String price) {
 17:         this.price = price;
 18:     }
 19:  
 20:     public Object clone() {
 21:         Object clone = null;
 22:         try {
 23:             clone = super.clone();
 24:         } catch (CloneNotSupportedException e) {
 25:             e.printStackTrace();
 26:         }
 27:         return clone;
 28:     }
 29: }

然后我们根据这个抽象类创建两个具体的原型类,如果用简单工厂方法的视角那就是一个具体产品类,不过这并不要紧:

  1: public class Book extends Product {
  2:     private String author;
  3:  
  4:     public String getAuthor() {
  5:         return author;
  6:     }
  7:     public void setAuthor(String author) {
  8:         this.author = author;
  9:     }
 10: } 
  1: public class Cloths extends Product {
  2:     private int size;
  3:  
  4:     public int getSize() {
  5:         return size;
  6:     }
  7:  
  8:     public void setSize(int size) {
  9:         this.size = size;
 10:     }
 11: }

现在我们创建一个工厂(为了演示多个具体产品类,这个工厂定义的比较特殊——生产书和衣服)来创建这些产品:

  1: public class ProductFactory {
  2:     private static Hashtable productMap = new Hashtable();
  3:  
  4:     public static Product getProduct(String productCode) {
  5:         Product cachedProduct = (Product) productMap.get(productCode);
  6:         return (Product) cachedProduct.clone();
  7:     }
  8:  
  9:     public static void loadCache() {
 10:         Book book = new Book();
 11:         book.setTitle("Java best practices");
 12:         book.setAuthor("Prashant");
 13:         book.setPrice("100");
 14:         productMap.put(book.getTitle(), book);
 15:         Cloths cloths = new Cloths();
 16:         cloths.setTitle("Jeans");
 17:         cloths.setSize(32);
 18:         cloths.setPrice("450");
 19:         productMap.put(cloths.getTitle(),cloths);
 20:     }
 21: }

这个工厂的特殊之处是需要复制的对象已经创立好了,在创建工厂实例的时候通过loadCache就可以完成。

下边我们使用这个工厂克隆生产一本书和一件衣服:

  1: public class ProductOutlet {
  2:     public static void main(String[] args) {
  3:         ProductFactory.loadCache();
  4:         Book clonedBook = (Book) ProductFactory.getProduct("Java best practices");
  5:         System.out.println("Product cloned as Book");
  6:         System.out.println("Title = " + clonedBook.getTitle());
  7:         System.out.println("Author = " + clonedBook.getAuthor());
  8:         System.out.println("Price = " + clonedBook.getPrice());
  9:         System.out.println("Product cloned as Cloths");
 10:         Cloths clonedCloths = (Cloths) ProductFactory.getProduct("Jeans");
 11:         System.out.println("Title = " + clonedCloths.getTitle());
 12:         System.out.println("Size = " + clonedCloths.getSize());
 13:         System.out.println("Price = " + clonedCloths.getPrice());
 14:     }
 15: }
3.UML总结

Prototype_2

这里没有画简单工厂模式的实体。

原型模式多用于创建复杂的或者耗时的实例,因为这种情况下,复制一个已经存在的实例使程序运行更高效;或者创建值相等,只是命名不一样的同类数据(克隆以后进行定制)。




本文转自gnuhpc博客园博客,原文链接:http://www.cnblogs.com/gnuhpc/archive/2012/12/21/2827653.html,如需转载请自行联系原作者

相关文章
|
设计模式 Java 数据库连接
【设计模式】【创建型模式】工厂方法模式(Factory Methods)
一、入门 什么是工厂方法模式? 工厂方法模式(Factory Method Pattern)是一种创建型设计模式,它定义了一个用于创建对象的接口,但由子类决定实例化哪个类。工厂方法模式使类的实例化延迟
355 16
|
设计模式 JavaScript Java
【设计模式】【创建型模式】原型模式(Prototype)
一、入门 什么是原型模式? 原型模式(Prototype Pattern)是一种创建型设计模式,它通过复制现有对象来创建新对象,而不是通过实例化类。 原型模式的核心是克隆(Clone),即通过复制现有
306 15
|
设计模式 负载均衡 监控
并发设计模式实战系列(2):领导者/追随者模式
🌟 ​大家好,我是摘星!​ 🌟今天为大家带来的是并发设计模式实战系列,第二章领导者/追随者(Leader/Followers)模式,废话不多说直接开始~
353 0
|
设计模式 监控 Java
并发设计模式实战系列(1):半同步/半异步模式
🌟 ​大家好,我是摘星!​ 🌟今天为大家带来的是并发设计模式实战系列,第一章半同步/半异步(Half-Sync/Half-Async)模式,废话不多说直接开始~
500 0
|
设计模式 安全 Java
并发设计模式实战系列(12):不变模式(Immutable Object)
🌟 大家好,我是摘星!🌟今天为大家带来的是并发设计模式实战系列,第十二章,废话不多说直接开始~
290 0
|
设计模式 算法 Java
设计模式觉醒系列(04)策略模式|简单工厂模式的升级版
本文介绍了简单工厂模式与策略模式的概念及其融合实践。简单工厂模式用于对象创建,通过隐藏实现细节简化代码;策略模式关注行为封装与切换,支持动态替换算法,增强灵活性。两者结合形成“策略工厂”,既简化对象创建又保持低耦合。文章通过支付案例演示了模式的应用,并强调实际开发中应根据需求选择合适的设计模式,避免生搬硬套。最后推荐了JVM调优、并发编程等技术专题,助力开发者提升技能。
|
设计模式 Prometheus 监控
并发设计模式实战系列(20):扇出/扇入模式(Fan-Out/Fan-In)(完结篇)
🌟 大家好,我是摘星!🌟今天为大家带来的是并发设计模式实战系列,第二十章,废话不多说直接开始~
399 0
|
设计模式 Java 关系型数据库
设计模式:工厂方法模式(Factory Method)
工厂方法模式是一种创建型设计模式,通过将对象的创建延迟到子类实现解耦。其核心是抽象工厂声明工厂方法返回抽象产品,具体工厂重写该方法返回具体产品实例。适用于动态扩展产品类型、复杂创建逻辑和框架设计等场景,如日志记录器、数据库连接池等。优点包括符合开闭原则、解耦客户端与具体产品;缺点是可能增加类数量和复杂度。典型应用如Java集合框架、Spring BeanFactory等。
|
设计模式 前端开发 搜索推荐
前端必须掌握的设计模式——模板模式
模板模式(Template Pattern)是一种行为型设计模式,父类定义固定流程和步骤顺序,子类通过继承并重写特定方法实现具体步骤。适用于具有固定结构或流程的场景,如组装汽车、包装礼物等。举例来说,公司年会节目征集时,蜘蛛侠定义了歌曲的四个步骤:前奏、主歌、副歌、结尾。金刚狼和绿巨人根据此模板设计各自的表演内容。通过抽象类定义通用逻辑,子类实现个性化行为,从而减少重复代码。模板模式还支持钩子方法,允许跳过某些步骤,增加灵活性。
935 11
|
设计模式 安全 Java
Kotlin教程笔记(51) - 改良设计模式 - 构建者模式
Kotlin教程笔记(51) - 改良设计模式 - 构建者模式