设计模式-工厂方法模式

基于面向对象语言开发中,免不得需要创建对象。前面讲解的"单例模式"也是如此,不过是要创建唯一的对象。本文要讲述“工厂方法模式”是要封装创建对象的过程。工厂,也称之为“制造厂”,用于创建具体的产品直接提供给外界使用。其实,这就是对产品的创建过程的封装,外界需要产品,那就通过工厂提供的途径获取。反映到代码世界中,如果对象的创建过程复杂,需要封装对象创建过程时,我们可不通过一个“工厂”来封装这个过程呢。工厂的具体代码表现形式要看封装的粒度。这里封装的粒度包括方法(代码块)、类。这二者也分别代表着简单工厂方法模式、工厂方法模式

一、简单工厂模式

简单工厂模式,即是对象创建过程的封装粒度为方法或代码块(一般来说是方法)。如下为一个创建汽车对象的工厂示例:

/**
 * 汽车(抽象)接口
 */
public interface Car {
    void start();

    void stop();
}
/**
 * 普通汽车实现类
 */
public class CommonCar implements Car{
    @Override
    public void start() {
        System.out.println("SUV 启动...");
    }

    @Override
    public void stop() {
        System.out.println("SUV 停止...");
    }
}
/**
 * 创建汽车对象的工厂
 */
public class CarFactory {

    public static Car createCarFactory() {
        return new CommonCar();
    }
}
public class Client {
    public static void main(String[] args) {
        // 获取对象
        Car caInstance = CarFactory.createCarFactory(); // 封装了所需对象创建的过程

        // 执行对象活动
        caInstance.start();
        caInstance.stop();
    }
}

在这里插入图片描述

简单工厂模式的实现方法及类图如上,用户端只需要通过CarFactory调用对应的方法即可获取对象,不需要感知到对象的创建过程。在大部分的业务代码中,还真就这种模式使用居多,比如某数据库客户端对象的获取、某加密算法对象获取等。但是,这么模式仅使用于比较简单的场景,如果业务上需要有多种数据库客户端、多种加密算法时,简单工厂方法模式就不能很好的胜任了。比如,上述场景的汽车对象会区分SUV及小轿车。那工厂方法模式就必须得改动,如改动为:

public static Car createCarFactory(String type) {
        if (type.equalsIgnoreCase("sedan")) {
            return new Sedan();     // 创建轿车的逻辑
        } else if (type.equalsIgnoreCase("suv")) {
            return new SUV();       // 创建SUV的逻辑
        } else {
            throw new IllegalArgumentException("无效的汽车类型" + type);
        }
    }

这样一改就有两个缺点:① 工厂方法逻辑完全改变了,也需要求客户端(上游依赖方)跟着改动;② 随着汽车种类的增多,if判断条件也会增加。多种类型汽车初始化逻辑存在于一个方法内部,非常不利于代码维护、调试。因此简单工厂模式的最大缺点就是不易扩展。进一步说,不易扩展的根因就在于将对象的创建逻辑封装与单一方法内部,不符合单一职责原则,也肯定不满足开闭原则了。
简单工厂模式的优点:

  • 避免对象创建过程的暴露
  • 实现简单,实用性高。
  • 适合创建对象种类少、规则相对稳定的业务。

简单工厂模式的优点:

  • 不符合单一职责原则,工厂创建对象的方法职责过重。
  • 不符合开闭原则,增加对象种类,需要改动原有工厂方法,甚至需要上游跟随改动。

二、工厂方法模式

为应用多种对象创建的需求,设计的工厂模式必须符合开闭原则。基于依赖倒置设计原则,工厂模块也应该有其抽象类,工厂与产品之间通过抽象类进行相互依赖。对于不同的抽象类用于创建不同的产品实例,因此,客户端仅需要感知创建那种对象使用那种工厂,和生活中例子一样,需要造汽车就找汽车工厂,需要衣服就找制服厂等。
还是以上述例子为例,如何实现SUV汽车、轿车的工厂,代码如下:
① 汽车接口

public interface Car {
    void start();

    void stop();
}

② 汽车具体实现类

public class SUV implements Car{
    @Override
    public void start() {
        System.out.println("SUV 启动...");
    }

    @Override
    public void stop() {
        System.out.println("SUV 停止...");
    }
}

public class Sedan implements Car{
    @Override
    public void start() {
        System.out.println("轿车 启动...");
    }

    @Override
    public void stop() {
        System.out.println("轿车 启动...");
    }
}

③ 工厂接口

public interface Factory {
    Car createCarInstance();
}

④ 工厂具体实现类

public class SUVFactory implements Factory{
    @Override
    public Car createCarInstance() {
        return new SUV();
    }
}

public class SedanFactory implements Factory{
    @Override
    public Car createCarInstance() {
        return new Sedan();
    }
}

⑤ 客户端

public class Client {
    public static void main(String[] args) {
        SUVFactory suvFactory = new SUVFactory();
        Car car4SUV = suvFactory.createCarInstance();   // 创建SUV汽车实例

        SedanFactory sedanFactory = new SedanFactory();
        Car car4Sedan = sedanFactory.createCarInstance();   // 创建SUV汽车实例
    }
}

在这里插入图片描述
工厂方法模式的示例代码及类图如上所示,这种设计模式解耦了工厂与产品的关系。高层与底层模块的关系符合依赖倒置原则,类之间的关系也符合迪米特法则。不同具体工厂类负责不同产品的实例创建,因此符合单一职责原则。从后续业务发展来看,即使后续增加产品类型,也只是增加新的产品类及对应的产品工厂类,不会改变已有的业务逻辑,因此这个设计方案是符合开闭原则的。
工厂方法模式的优点:

  • 避免对象创建过程的暴露
  • 符合开闭原则,易扩展
  • 适合创建对象种类多、规则相对不稳定的业务。

工厂方法模式的缺点:

  • 实现负责,容易造成类的激增。

三、后注

本篇文章讲解了简单工厂模式以及工厂方法模式,这里我想说下工厂设计模式的宏观理解。简单的创建对象流程即使客户端直接创建对象,然后使用。
在这里插入图片描述
这个是最简单的创建并使用对象的方法,也是最常见的。但是问题在于,如果对象的创建过程及其复杂,难道要所有的调用方均复制一遍这块代码,增加了系统代码的臃肿程度。
因此,对于创建过程我们可以使用一种方式封装起来。封装在面向对象设计中就是为了复用代码,复用这套复杂创建逻辑的代码。封装的究极解决方案就是“加一层”,也是大部分情况都能很好解决问题的方案。何为加一层?就是在客户端与要创建对象的类之间加一层,将创建&初始化对象的逻辑封装到中间层-工厂。之后客户端要创建对象就找工厂,将客户端与对象复杂的创建过程进行解耦。【这里的基本思路为是谁导致的问题,找个人来负责解决它,即是重要逻辑细化】【如何更好使用单一职责原则?那块逻辑对业务影响大,那就需要一个方法、类或模块来负责之】
因此,创建对象这件事就仅是客户端和目标类之间的关系,中间有个工厂需要介入。
在这里插入图片描述
负责封装目标对象的工厂模式的具体表现要根据目标对象的复杂程度而定,复杂程度包括对象的种类数、对象业务规则的稳定性来确定。如果要创建的对象十分简单,那就使用简单工厂模式即可,反之则需要使用工厂方法模式来保证可扩展性。

提问:会不会存在工厂也比较多的时候呢?如何解决?

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值