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,是同一产品族