设计模式二: 工厂方法(Factory Method)

简介

工厂方法模式是创建型模式的一种, 核心结构有四个角色: 抽象工厂,具体工厂,抽象产品,具体产品;

实现层面上,该模式定义一个创建产品的接口,将实际创建工作推迟到具体工厂类实现, 一个产品对应一个工厂, 这样的好处是当有新产品引入时可以不修改具体的工厂角色.

意图

定义了一个创建对象的接口,但由子类决定要实例化哪个类。工厂方法把实例化操作推迟到子类。

类图

设计模式二: 工厂方法(Factory Method)

实现

以汽车举例. 按照本模式核心的四个角色分别定义, 并增加模式调用角色(此处顺便提一下简单工厂模式, 比较简单, 具体实现参看代码).

  1. 抽象产品, 定义接口 Car
/**
 * 汽车接口
 */
public interface Car {

    /**
     * 开车
     */
    void run();

}
  1. 抽象工厂 CarFactory
/**
 * 汽车工厂接口
 */
public interface CarFactory {

    /**
     * 生产汽车
     * @return 实现Car接口的类实例
     */
    Car getCar();

}
  1. 具体产品 BMWCar,QQCar
/**
 * BMW 汽车实现类
 */
public class BMWCar implements Car {

    public void run() {
        System.out.println("BMW run...");
    }

}

/**
 * QQ 汽车实现类
 */
public class QQCar implements Car {

    public void run() {
        System.out.println("QQ run...");
    }

}
  1. 具体工厂 BMWFactory,QQFactory
/**
 * BMW 汽车工厂实现
 */
public class BMWFactory implements CarFactory {

    public Car getCar() {
        return new BMWCar();
    }

}

/**
 * QQ 汽车工厂实现
 */
public class QQFactory implements CarFactory {

    public Car getCar() {
        return new QQCar();
    }

}
  1. 调用类 Client
/**
 * 调用方角色
 */
public class Client {

    /**
     * 调用演示
     */
    public void drive(){
        CarFactory bmwFactory = new BMWFactory();
        Car bmwCar = bmwFactory.getCar();
        bmwCar.run();

        CarFactory qqFactory = new QQFactory();
        Car qqCar = qqFactory.getCar();
        qqCar.run();

        Car car = getCarByType(1);
        if (car!=null){
            car.run();
        }
        car = getCarByType(2);
        if (car!=null){
            car.run();
        }
    }

    /**
     * 简单工厂实现, 通过传入的参数返回具体的汽车
     * @param carType
     * @return
     */
    public Car getCarByType(int carType){

        Car car = null;
        switch (carType){
            case 1:
                CarFactory bmwFactory = new BMWFactory();
                Car bmwCar = bmwFactory.getCar();
                car = bmwCar;
                break;
            case 2:
                CarFactory qqFactory = new QQFactory();
                Car qqCar = qqFactory.getCar();
                car = qqCar;
                break;
        }

        return car;
    }

}

总结

优点: 1. 隐藏产品创建细节,我们只需要关系具体产品对应的具体工厂; 2. 完全符合设计模式"开闭"原则,当有新产品加入时只需要实现对应的工厂类;

缺点: 产品种类太多时维护难度增加.

上一篇:设计模式三: 代理模式(Proxy) -- JDK的实现方式


下一篇:第三章 Java的基础程序设计结构