【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,如需转载请自行联系原作者

相关文章
|
14天前
|
设计模式 前端开发 搜索推荐
前端必须掌握的设计模式——模板模式
模板模式(Template Pattern)是一种行为型设计模式,父类定义固定流程和步骤顺序,子类通过继承并重写特定方法实现具体步骤。适用于具有固定结构或流程的场景,如组装汽车、包装礼物等。举例来说,公司年会节目征集时,蜘蛛侠定义了歌曲的四个步骤:前奏、主歌、副歌、结尾。金刚狼和绿巨人根据此模板设计各自的表演内容。通过抽象类定义通用逻辑,子类实现个性化行为,从而减少重复代码。模板模式还支持钩子方法,允许跳过某些步骤,增加灵活性。
|
2月前
|
设计模式 安全 Java
Kotlin教程笔记(51) - 改良设计模式 - 构建者模式
Kotlin教程笔记(51) - 改良设计模式 - 构建者模式
|
2月前
|
设计模式 开发者 Python
Python编程中的设计模式:工厂方法模式###
本文深入浅出地探讨了Python编程中的一种重要设计模式——工厂方法模式。通过具体案例和代码示例,我们将了解工厂方法模式的定义、应用场景、实现步骤以及其优势与潜在缺点。无论你是Python新手还是有经验的开发者,都能从本文中获得关于如何在实际项目中有效应用工厂方法模式的启发。 ###
|
2月前
|
设计模式 安全 Java
Kotlin - 改良设计模式 - 构建者模式
Kotlin - 改良设计模式 - 构建者模式
|
2月前
|
设计模式 安全 Java
Kotlin教程笔记(51) - 改良设计模式 - 构建者模式
Kotlin教程笔记(51) - 改良设计模式 - 构建者模式
50 1
|
3月前
|
设计模式 Java Kotlin
Kotlin学习笔记 - 改良设计模式 - 迭代器模式
Kotlin学习笔记 - 改良设计模式 - 迭代器模式
39 2
|
3月前
|
设计模式 JavaScript Scala
Kotlin学习笔记 - 改良设计模式 - 责任链模式
Kotlin学习笔记 - 改良设计模式 - 责任链模式
49 0
|
3月前
|
设计模式 Java Kotlin
Kotlin 学习笔记- 改良设计模式 - 装饰者模式
Kotlin 学习笔记- 改良设计模式 - 装饰者模式
37 0
|
4月前
|
设计模式 数据库连接 PHP
PHP中的设计模式:提升代码的可维护性与扩展性在软件开发过程中,设计模式是开发者们经常用到的工具之一。它们提供了经过验证的解决方案,可以帮助我们解决常见的软件设计问题。本文将介绍PHP中常用的设计模式,以及如何利用这些模式来提高代码的可维护性和扩展性。我们将从基础的设计模式入手,逐步深入到更复杂的应用场景。通过实际案例分析,读者可以更好地理解如何在PHP开发中应用这些设计模式,从而写出更加高效、灵活和易于维护的代码。
本文探讨了PHP中常用的设计模式及其在实际项目中的应用。内容涵盖设计模式的基本概念、分类和具体使用场景,重点介绍了单例模式、工厂模式和观察者模式等常见模式。通过具体的代码示例,展示了如何在PHP项目中有效利用设计模式来提升代码的可维护性和扩展性。文章还讨论了设计模式的选择原则和注意事项,帮助开发者在不同情境下做出最佳决策。
|
3月前
|
设计模式 Java Kotlin
Kotlin教程笔记(51) - 改良设计模式 - 构建者模式
本教程详细讲解Kotlin语法,适合希望深入了解Kotlin的开发者。对于快速学习Kotlin语法,推荐查看“简洁”系列教程。本文重点介绍了构建者模式在Kotlin中的应用与改良,包括如何使用具名可选参数简化复杂对象的创建过程,以及如何在初始化代码块中对参数进行约束和校验。
35 3