1、简单工厂(Simple Factory)

概述:

在创建一个对象时不向客户暴露内部细节,并提供一个创建对象的通用接口。

错误的方式:





 

正确的方式:


 

 

简单工厂把实例化的操作单独放到一个类中,这个类就成为简单工厂类,让简单工厂类来决定应该用哪个具体子类来实例化。

这样做能把客户类和具体子类的实现解耦,客户类不再需要知道有哪些子类以及应当实例化哪个子类。客户类往往有多个,如果不使用简单工厂,那么所有的客户类都要知道所有子类的细节。而且一旦子类发生改变,例如增加子类,那么所有的客户类都要进行修改。

注:并不是GOF23种设计模式

In JDK:

Calendar

-createCalendar()

 

2、工厂方法(Factory Method

概述:

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

public abstract class Factory {

    abstract public Product factoryMethod();

    public void doSomething() {

    Product product = factoryMethod();

    // do something with the product

    }

}

public class ConcreteFactory extends Factory {

    public Product factoryMethod() {

        return new ConcreteProduct();

    }

}

public class ConcreteFactory1 extends Factory {

    public Product factoryMethod() {

        return new ConcreteProduct1();

    }

}

public class ConcreteFactory2 extends Factory {

    public Product factoryMethod() {

        return new ConcreteProduct2();

    }

}

在简单工厂中,创建对象的是另一个类,而在工厂方法中,是由子类来创建对象。

上图中,Factory 有一个 doSomething() 方法,这个方法需要用到一个产品对象,这个产品对象由 factoryMethod() 方法创建。该方法是抽象的,需要由子类去实现。

使用场景:

创建对象需要大量重复的代码

客户端( 应用层)不需要依赖产品类实例如何创建、实现等细节

一个类通过其子类来指定创建哪个对象

优点:

用户只需要关心所需产品对应的工厂,无须关心创建细节 加入新产品符合开闭原则,提高可扩展性

缺点:

类的个数容易过多,增加复杂度 增加了系统的抽象性和理解难度

 

In JDK

Collection 

~iterator()

ArrayList

+iterator()

3、抽象工厂(Abstract Factory)

概述:

适用场景:

客户端(应用层)不依赖于产品类实例如何被创建、实现等细节

强调一系列相关的产品对象(属于同一产品族)一起使用创建对象需要大量重复的代码

提供一个产品类的库,所有的产品以同样的接口出现,从而使客户端不依赖于具体实现

优点:

具体产品在应用层代码隔离,无须关心创建细节,将一个系列的产品族统一到一起创建

缺点:

规定了所有可能被创建的产品集合,产品族中扩展新的产品困难,需要修改抽象工厂的接口

增加了系统的抽象性和理解难度

关于抽象工厂的产品族:

提供一个接口,用于创建相关的对象家族 

UML&实例DEMO

源码链接:https://github.com/NoSuchClass/design_pattern/tree/master/src/creational/abstractfactory

 

抽象工厂模式创建的是产品族,也就是很多对象而不是一个对象,并且这些对象是相关的,也就是说必须一起创建出来。而工厂方法模式只是用于创建一个对象,这和抽象工厂模式有很大不同。

抽象工厂模式用到了工厂方法模式来创建单一对象,AbstractFactory 中的 createProductA() 和 createProductB() 方法都是让子类来实现,这两个方法单独来看就是在创建一个对象,这符合工厂方法模式的定义。

至于创建对象的家族这一概念是在 Client 体现,Client 要通过 AbstractFactory 同时调用两个方法来创建出两个对象,在这里这两个对象就有很大的相关性,Client 需要同时创建出这两个对象。

从高层次来看,抽象工厂使用了组合,即 Cilent 组合了 AbstractFactory,而工厂方法模式使用了继承。

 

In JDK

java.sql.Connection:

这就是一个抽象工厂,里面都是一个产品族的获取

java.sql.Statement

In MyBatis

java.sql.SqlSessionFactory:

这儿不但获取了SqlSession,还获取了Configuration,是同一产品族