`
derekop
  • 浏览: 22600 次
  • 性别: Icon_minigender_1
  • 来自: 天津
社区版块
存档分类
最新评论

在JAVA中应用设计模式FactoryMethod

    博客分类:
  • java
阅读更多

 FactoryMethod是一种创建性模式,它定义了一个创建对象的接口,但是却让子类来决定具体实例化哪一个类.当一个类无法预料要创建哪种类的对象或是一个类需要由子类来指定创建的对象时我们就需要用到Factory Method 模式了.简单说来,Factory Method可以根据不同的条件产生不同的实例,当然这些不同的实例通常是属于相同的类型,具有共同的父类.Factory Method把创建这些实例的具体过程封装起来了,简化了客户端的应用,也改善了程序的扩展性,使得将来可以做最小的改动就可以加入新的待创建的类. 通常我们将Factory Method作为一种标准的创建对象的方法,当发现需要更多的灵活性的时候,就开始考虑向其它创建型模式转化:

  图1是Factory Method 模式的结构图,这里提供了一些术语,让我们可以进行更方便的描述:

  Product: 需要创建的产品的抽象类.

  ConcreteProduct: Product的子类,一系列具体的产品.

  Creator: 抽象创建器接口,声明返回Product类型对象的Factory Method.

  ConcreteCreator: 具体的创建器,重写Creator中的Factory Method,返回ConcreteProduct类型的实例.
 
图1: Factory Method 模式结构

 

 

 由此可以清楚的看出这样的平行对应关系: Product <====> Creator ; ConreteProduct <====> ConreteCreator

  抽象产品对应抽象创建器,具体产品对应具体创建器.这样做的好处是什么呢?为什么我们不直接用具体的产品和具体的创建器完成需求呢?实际上我们也可以这样做.但通过Factory Method模式来完成,客户(client)只需引用抽象的Product和Creater,对具体的ConcreteProduct和 ConcreteCreator可以毫不关心,这样做我们可以获得额外的好处:

  首先客户端可以统一从抽象创建器获取产生的实例,Creator的作用将client和产品创建过程分离开来,客户不用操心返回的是那一个具体的产品,也不用关心这些产品是如何创建的.同时,ConcreteProduct也被隐藏在Product后面,ConreteProduct继承了Product的所有属性,并实现了 Product中定义的抽象方法,按照Java中的对象造型(cast)原则,通过ConcreteCreator产生的ConcreteProduct 可以自动的上溯造型成Product.这样一来,实质内容不同的ConcreteProduct就可以在形式上统一为Product,通过Creator 提供给client来访问.

  其次,当我们添加一个新的ConcreteCreator时,由于Creator所提供的接口不变,客户端程序不会有丝毫的改动,不会带来动一发而牵全身的灾难, 这就是良好封装性的体现.但如果直接用ConcreteProduct和ConcreteCreator两个类是无论如何也做不到这点的. 优良的面向对象设计鼓励使用封装(encapsulation)和委托(delegation),而Factory Method模式就是使用了封装和委托的典型例子,这里封装是通过抽象创建器Creator来体现的,而委托则是通过抽象创建器把创建对象的责任完全交给具体创建器ConcreteCreator来体现的.

  现在,请再回头看看基本概念中的那段话,开始也许觉得生涩难懂,现在是不是已经明朗化了很多.

  下面让我们看看在 Java 中如何实现Factory Method模式,进一步加深对它的认识.

 

先说明一点,用Factory Method模式创建对象并不一定会让我们的代码更短,实事上往往更长,我们也使用了更多的类,真正的目的在于这样可以灵活的,有弹性的创建不确定的对象.而且,代码的可重用性提高了,客户端的应用简化了,客户程序的代码会大大减少,变的更具可读性.

  标准实现: 这里我采用Bruce Eckel 用来描述OO思想的经典例子 Shape.这样大家会比较熟悉一些.我完全按照图1中所定义的结构写了下面的一段演示代码.这段代码的作用是创建不同的Shape实例,每个实例完成两个操作:draw和erase.具体的创建过程委托?oShapeFactory来完成.

  1.a 首先定义一个抽象类Shape,定义两个抽象的方法.

 

  abstract class Shape {
   // 勾画shape
   public abstract void draw();
   // 擦去 shape
   public abstract void erase();
   public String name;

   public Shape(String aName){

   name = aName;
   }
   }

 

   1.b 定义 Shape的两个子类: Circle, Square,实现Shape中定义的抽象方法

 // 圆形子类
   class Circle extends Shape {
   public void draw() {
   System.out.println("It will draw a circle.");
   }

   public void erase() {
   System.out.println("It will erase a circle.");

   }

   // 构造函数
   public Circle(String aName){

   super(aName);

   }
   }
   // 方形子类
   class Square extends Shape {
   public void draw() {
   System.out.println("It will draw a square.");
   }

   public void erase() {
   System.out.println("It will erase a square.");
   }
   // 构造函数
   public Square(String aName){

   super(aName);
   }
   }

  1.c 定义抽象的创建器,anOperation调用factoryMethod创建一个对象,并对该对象进行一系列操作.

 abstract class ShapeFactory {
   protected abstract Shape factoryMethod(String aName);
   // 在anOperation中定义Shape的一系列行为
   public void anOperation(String aName){

   Shape s = factoryMethod(aName);

   System.out.println("The current shape is: " + s.name);
   s.draw();

   s.erase();
   }
   }

  1.d 定义与circle和square相对应的两个具体创建器CircleFactory,SquareFactory,实现父类的methodFactory方法

 // 定义返回 circle 实例的 CircleFactory
   class CircleFactory extends ShapeFactory {
   // 重载factoryMethod方法,返回Circle对象
   protected Shape factoryMethod(String aName) {
   return new Circle(aName + " (created by CircleFactory)");

   }
   }
   // 定义返回 Square 实例的 SquareFactory
   class SquareFactory extends ShapeFactory {
   // 重载factoryMethod方法,返回Square对象

   protected Shape factoryMethod(String aName) {
   return new Square(aName + " (created by SquareFactory)");
   }
   }

  1.e 测试类:请注意这个客户端程序多么简洁,既没有罗嗦的条件判断语句,也无需关心ConcreteProduct和ConcreteCreator的细节 (因为这里我用anOperation封装了Product里的两个方法,所以连Product的影子也没看见,当然把Product里方法的具体调用放到客户程序中也是不错的)

 class Main {

   public static void main(String[] args){

   ShapeFactory sf1 = new SquareFactory();
   ShapeFactory sf2 = new CircleFactory();
   sf1.anOperation("Shape one");

   sf2.anOperation("Shape two");
   }
   }

  运行结果如下:

 The current shape is: Shape one (created by SquareFactory)
  It will draw a square.
   It will erase a square.
   The current shape is: Shape two (created by CircleFactory)
   It will draw a circle.
   It will erase a circle.

 

 参数化的Factory Method: 这种方式依靠指定的参数作为标志来创建对应的实例,这是很常见的一种办法.比如JFC中的BorderFactory就是个很不错的例子. 以下的这个例子是用字符串作为标记来进行判断的,如果参数的类型也不一样,那就可以用到过载函数来解决这个问题,定义一系列参数和方法体不同的同名函数, 这里java.util.Calendar.getInstance()又是个极好的例子.参数化的创建方式克服了Factory Method模式一个最显著的缺陷,就是当具体产品比较多时,我们不得不也建立一系列与之对应的具体构造器. 但是在客户端我们必须指定参数来决定要创建哪一个类.

  2.a 我们在第一种方法的基础上进行修改,首先自定义一个的异常,这样当传入不正确的参数时可以得到更明显的报错信息.

 class NoThisShape extends Exception {
   public NoThisShape(String aName) {
   super(aName);
   }
   }

 

2.b去掉了ShapeFactory的两个子类,改为由ShapeFactory直接负责实例的创建. ShapeFactory自己变成一个具体的创建器,直接用参数化的方法实现factoryMethod返回多种对象.

 abstract class ShapeFactory {

   private static Shape s;

   private ShapeFactory() {}
   static Shape factoryMethod(String aName, String aType) throws NoThisShape{
  if (aType.compareTo("square")==0)

   return new Square(aName);
   else if (aType.compareTo("circle")==0)
   return new Circle(aName);

   else throw new NoThisShape(aType);

   }
   // 在anOperation中定义Shape的一系列行为
   static void anOperation(String aName, String aType) throws NoThisShape{

   s = factoryMethod(aName, aType);

   System.out.println("The current shape is: " + s.name);
   s.draw();

   s.erase();

   }

   }

  2.c 测试类:这里客户端必须指定参数来决定具体创建哪个类.这个例子里的anOperation是静态函数,可以直接引用.

  class Main {
   public static void main(String[] args) throws NoThisShape{
   ShapeFactory.anOperation("Shape one","circle");
   ShapeFactory.anOperation("Shape two","square");

   ShapeFactory.anOperation("Shape three", "delta");

   }

   }

 

  运行结果如下:

  class Main {

   public static void main(String[] args) throws NoThisShape{
  ShapeFactory.anOperation("Shape one","circle");
   ShapeFactory.anOperation("Shape two","square");
   ShapeFactory.anOperation("Shape three", "delta");

   }

   }

 

 动态装载机制:

  有的时候我们会把ConcreteProduct的实例传给创建器作为参数,这种情况下,如果在创建器里完成创建过程,就必须判断参数的具体类型(用instanceof),然后才能产生相应的实例,那么比较好的做法是利用Java的动态装载机制来完成这件事.比如:

  我们得到一个Shape的子类s,但不知道具体是那个子类,就可以利用Class类自带的方法newInstance()得到实例

  return (Shape)s.getClass().newInstance();

  

 

转自 http://tech.it168.com/a2009/0221/266/000000266213_1.shtml

分享到:
评论

相关推荐

    在Java中应用设计模式-FactoryMethod

    在设计模式中,FactoryMethod也是比较简单的一个,但应用非常广泛,EJB,RMI,COM,CORBA,Swing中都可以看到此模式的影子,它是最重要的模式之一.在很多地方我们都会看到xxxFactory这样命名的类,那么,什么是FactoryMethod,...

    java 23种设计模式及具体例子

    java 中的 23 种设计模式可以分为三大类:创建型模式、结构型模式、行为型模式。每种模式都有其特点和应用场景。 1. 工厂方法模式(Factory Method) 工厂方法模式分为三种:普通工厂模式、多个工厂方法模式、静态...

    设计模式_java语言中的应用

    主要内容有:Iterator-迭代器、Adapter(适配器)、Template Method(模板方法)、Factory Method(工厂方法)、Abstract Factory(抽象工厂)等。

    Java 23种设计模式全归纳

    本资源包汇总了Java中23种经典设计模式的详细介绍和使用示例,旨在帮助开发者系统性地学习和掌握设计模式在软件开发中的应用。这些设计模式分为三大类:创建型模式、结构型模式和行为型模式,每种模式都包括定义、...

    Java 23种设计模式通解

    设计模式是软件工程的基石,它们被广泛应用于软件开发中,以提高代码的可重用性、可读性和可维护性。 设计模式的分类: * 创建型模式(5种):工厂方法模式、抽象工厂模式、单例模式、建造者模式、原型模式 * 结构...

    java设计模式示例

    1.工厂方法模式(Factory Method) 将程序中创建对象的操作,单独出来处理,创建一个产品的工厂接口,把实际的工作转移到具体的子类。大大提高了系统扩展的柔性,接口的抽象化处理给相互依赖的对象创建提供了最好的...

    Java23种设计模式

    "Java23种设计模式" ...Java设计模式的应用非常广泛,包括但不限于:数据库设计、网络编程、用户界面设计等领域。掌握Java设计模式可以帮助开发者更好地设计和实现软件系统,提高软件的可维护性、灵活性和可重用性。

    结城浩《图解设计模式》笔记

    本笔记主要讲述了设计模式的基本概念和应用,包括 Iterator 模式、Adapter 模式、TemplateMethod 模式、FactoryMethod 模式和 Singleton 模式等。通过对这些设计模式的详细解释和示例代码的分析,我们可以更好地理解...

    深入浅出设计模式(中文版电子版)

    3.2FactoryMethodPattern(工厂方法模式) 35 3.2.1定义 35 3.2.2现实例子——兵工厂 36 3.2.3C#实例——多文档系统 37 3.2.4Java实例——扩展了的手机工厂 41 3.2.5优势和缺陷 44 3.2.6应用情景 44 3.3...

    深入浅出设计模式(中文版)

    3.2FactoryMethodPattern(工厂方法模式) 35 3.2.1定义 35 3.2.2现实例子——兵工厂 36 3.2.3C#实例——多文档系统 37 3.2.4Java实例——扩展了的手机工厂 41 3.2.5优势和缺陷 44 3.2.6应用情景 44 3.3...

    JAVA设计模式之创造型模式介绍.doc

    在JAVA设计模式中,创造型模式包括工厂方法、抽象工厂、创造者模式、单态模式、原型模式等五类。 一、工厂方法模式 工厂方法模式定义一个用于创建对象的接口,让子类决定实例化哪一个类。FactoryMethod使一个类的...

    [源代码] 《易学 设计模式》 随书源代码

    ☆ 资料说明:☆ 《易学 设计模式》 郭志学 编著 随书源代码 ☆ 章节清单:☆ 第01章 欲速则不达:了解设计模式 第02章 磨刀不误砍柴工:UML语言概述 ...第27章 无招胜有招:如何在设计中应用设计模式

    Head First Design Patterns 英文版 Head First设计模式

    第1章到第11章陆续介绍的设计模式为Strategy、Observer、Decorator、Abstract Factory、Factory Method、Singleton,Command、Adapter、Facade、TemplateMethod、Iterator、Composite、State、Proxy。最后三章比较...

    研磨设计模式-part2

    《研磨设计模式》在内容上深入、技术上实用、和实际开发结合程度很高,书中大部分的示例程序都是从实际项目中简化而来,因此很多例子都可以直接拿到实际项目中使用。如果你想要深入透彻地理解和掌握设计模式,并...

    研磨设计模式-part4

    《研磨设计模式》在内容上深入、技术上实用、和实际开发结合程度很高,书中大部分的示例程序都是从实际项目中简化而来,因此很多例子都可以直接拿到实际项目中使用。如果你想要深入透彻地理解和掌握设计模式,并...

    研磨设计模式-part3

    《研磨设计模式》在内容上深入、技术上实用、和实际开发结合程度很高,书中大部分的示例程序都是从实际项目中简化而来,因此很多例子都可以直接拿到实际项目中使用。如果你想要深入透彻地理解和掌握设计模式,并...

    第14讲谈谈你知道的设计模式1

    在 Java 基础类库中,有很多典型的设计模式应用案例。例如,InputStream 是一个抽象类,标准类库中提供了FileInputStream、ByteArrayInputStream 等各种不同的子类,分别从不同角度对 InputStream 进行了功能扩展,...

    Head.First设计模式_PDF.part1

    第1章到第11章陆续介绍的设计模式为Strategy、Observer、Decorator、Abstract Factory、Factory Method、Singleton、Command、Adapter、Facade、Templat Method、Iterator、Composite、State、Proxy。最后三章比较...

    二十三种设计模式【PDF版】

    2.设计模式是比 J2EE 等框架软件更小的体系结构,J2EE 中许多具体程序都是应用设计模式来完成的,当你深入到 J2EE 的内 部代码研究时,这点尤其明显,因此,如果你不具备设计模式的基础知识(GoF 的设计模式),你很难...

Global site tag (gtag.js) - Google Analytics