`
sysutyphoon
  • 浏览: 5708 次
  • 性别: Icon_minigender_1
  • 来自: 广州
最近访客 更多访客>>
社区版块
存档分类
最新评论

工厂模式

阅读更多
    已经有很多介绍工厂模式的文章了。这里整理汇总下。

1、工厂模式的核心思想及分类

    工厂方法模式的作用是负责实例化同一个接口的多个类。工厂方法模式的意义是定义一个创建产品对象的工厂类,由该工厂统一创建继承了同一个接口的多个产品对象。

    工厂方法可以分为三个子类型:
   
  • 工厂方法模式(Factory Method):最基本的工厂模式;
  •    
  • 多个工厂方法模式:对工厂方法模式的扩展;
  •    
  • 简单工厂模式(Simple Factory):一种特殊的工厂模式,其工厂方法是静态的,因此也称为静态工厂方法模式

2、第一种——工厂方法模式

    工厂方法模式提供了一个工厂方法类,并提供了一个工厂方法函数,该方法负责创建所有的抽象产品对象。
    仍然使用互联网上广为流传的****户装B的例子:
    话说十年前,有一个****户,他家有三辆汽车——Benz奔驰、Bmw宝马、Audi奥迪,还雇了司机为他开车。不过,****户坐车时总是怪怪的:上 Benz车后跟司机说“开奔驰车!”,坐上Bmw后他说“开宝马车!”,坐上Audi说“开奥迪车!”。你一定说:这人有病!直接说开车不就行了?!
    而当把这个暴发户的行为放到我们程序设计中来时,会发现这是一个普遍存在的现象。幸运的是,这种有病的现象在OO(面向对象)语言中可以避免了。
   
//接口(抽象产品角色)
public interface Car{
      public void drive();
}

//具体产品角色
public class Benz implements Car{
      public void drive() {
         System.out.println("Driving Benz ");
      }
}

public class Bmw implements Car{
      public void drive() {
       System.out.println("Driving Bmw ");
      }
}
//...其他车的实现

//工厂方法类
public class Driver{
     //工厂方法.注意 返回类型为抽象产品角色
     public Car driveCar(String s)throws Exception{
     //判断逻辑,返回具体的产品角色给Client
     if(s.equalsIgnoreCase("Benz"))
           return new Benz();
     else if(s.equalsIgnoreCase("Bmw"))
           return new Bmw();
      ......   
     else throw new Exception();
     }
}

//使用:欢迎暴发户出场.....
public class Magnate{
     public static void main(String[] args){
        try{
          Driver driver = new Driver();
          //告诉司机我今天坐奔驰                    
          Car car = driver.driveCar("benz");
          //下命令:开车                           
          car.drive();
          ...
         }
      }


3、第二种——多个工厂方法模式
    对于第一种工厂方法模式。每次调用driveCar函数时必须传入一个类型参数。这种工厂类显得比较稚嫩,因为当传递参数错误时,就不能正确的创建产品。
    第二种工厂模式为工厂类提供多个工厂方法,分别创建不同的产品对象。
    //工厂方法类
public class Driver{
     public Car driveBmw(){
           return new Bmw();
     }
      
     public Car driveBenz(){
           return new Benz();
     }

    //....
}

//使用:欢迎暴发户出场.....
public class Magnate{
     public static void main(String[] args){
        try{
          Driver driver = new Driver();
          //告诉司机我今天坐奔驰                    
          Car car = driver.driveBenz();
          //下命令:开车                           
          car.drive();
          ...
         }
      }


4、第三种——静态工厂方法模式(简单工厂模式)
经过以上的改进,使得工厂类已经很完美了,但是还是美中不足。因为每一次创建实例需要使用工厂类时,都需要创建工厂类。此时可以使用静态工厂模式,将工厂方法设置为静态的。

public class Driver{
     public static Car driveBmw(){
           return new Bmw();
     }
      
     public static Car driveBenz(){
           return new Benz();
     }

    //....
}

//使用:欢迎暴发户出场.....
public class Magnate{
     public static void main(String[] args){
        try{
          //告诉司机我今天坐奔驰                    
          Car car = driver.driveBenz();
          //下命令:开车                           
          car.drive();
          ...
         }
      }


5、工厂方法模式中涉及的角色
   抽象工厂,具体工厂,抽象产品,具体产品。

6、工厂方法模式使用的前提
  • 有创建一批有相同接口对象的需求
  • 不想暴露太多类的细节给使用者,或者隐藏对象的创建工作
  • 产品的等级机构比较复杂,使用简单工厂模式可能会造成扩展性问题。
  • 当一个类不知道他必须创建的类的时候,或者一个类希望由他的子类制定他所创建的对象的时候。举个例子:系统框架使用抽象类定义,创建和维护对象之间的关系,但通常这些具体的类需要有客户端来具体化,对于框架无法知道具体子类的情况下,使用工厂模式来进行框架的开发维护。所以,工厂模式也被称作“虚拟构造器”

7、工厂方法模式的优点缺点
优点:
  • 优化了简单工厂模式,做到了“开-闭”原则。
  • 可以做到把具体的产品创建过程延迟的具体的子类工厂,使得基类工厂可以基于工厂方法创建的抽象对象工作。
缺点:
  • 暴露具体工厂,但多数情况这不会造成问题,尤其是对于虚拟构造器,就显得更加合理。


对于工厂方法模式子类创建的模式,有点让大家联想到模版模式,其主要区别是工厂方法模式主要目的是把创建工作推迟到子类,而模版模式则是把剩余的逻辑交给子类。这二者也可以结合起来使用。

   
  

   

分享到:
评论

相关推荐

Global site tag (gtag.js) - Google Analytics