在工厂方法模式中具体工厂负责生产具体的产品,每一个具体工厂对应一种具体产品,工厂方法具有唯一性,一般情况下,一个具体工厂中只有一个或者一组重载的工厂方法。但是有时候我们希望一个工厂可以提供多个产品对象,而不是单一的产品对象,如一个地图工厂,它可以生产院内地图、院外地图,而不是只生产一种地图。
-
产品等级结构
产品等级结构即产品的继承结构,如一个抽象类是院外地图,其子类百度院外地图、高德院外地图、广义恒鑫院外地图。则抽象院外地图与具体的院外地图之间构成了一个产品等级结构,抽象院外地图是父类,而具体的院外地图是其子类。
-
产品族
在抽象工厂模式中,产品族是指由同一个工厂生产的,位于不同产品等级结构中的一组产品,如百度地图工厂生产的百度院内地图、百度院外地图。那么百度院内地图位于院内地图等级结构中,百度院外地图位于院外地图等级结构中,百度院内地图、百度院外地图构成了一个产品族。
-
AbstractFactory(抽象工厂)
它声明了一组用于创建一族产品的方法,每一个方法对应一种产品。
public interface MapFactory {
GWIOutdoorMapBaseLayout createOutdoorMapView(Context context);
GWIIndoorMapBaseLayout createIndoorMapView(Context context);
}
-
ConcreteFactory(具体工厂)
它实现了在抽象工厂中声明的创建产品的方法,生成一组具体产品,这些产品构成了一个产品族,每一个产品都位于某个产品等级结构中。
public class BaiduMapFactory implements MapFactory {
@Override
public GWIOutdoorMapBaseLayout createOutdoorMapView(Context context) {
//返回百度地图的院外地图
}
@Override
public GWIIndoorMapBaseLayout createIndoorMapView(Context context) {
//返回百度地图的院内地图
}
}
-
AbstractProduct(抽象产品)
它为每种产品声明接口,在抽象产品中声明了产品所具有的业务方法。
public interface GWIOutdoorMapBaseLayout {
void display();
}
-
ConcreteProduct(具体产品)
它定义具体工厂生产的具体产品对象,实现抽象产品接口中声明的业务方法。
public class GWIBaiduOutdoorMapLayout extends GWIOutdoorMapBaseLayout {
@Override
public void display() {
//显示百度地图的院外地图
}
}
-
调用方式
通过创建一个工厂的实例,来得到对应的院内地图和院外地图。
MapFactory mapFactory = new BaiduMapFactory();
GWIBaiduOutdoorMapLayout gwiBaiduOutdoorMapLayout = mapFactory.createOutdoorMapView(this);
GWIBaiduIndoorMapLayout gwiBaiduIndoorMapLayout = mapFactory.createIndoorMapView(this);
gwiBaiduOutdoorMapLayout.display();
gwiBaiduIndoorMapLayout.display();
- 抽象工厂模式通过把产品划分为族的概念,来减少工厂类的个数,即一个工厂类不再是单一的生产一个产品而是生产一族产品,这一族产品必定会有个约束才能称作是一族,以上举的例子中,假如百度院内地图和百度院外地图需要同时显示,这个就是一个约束,所以可以把它们看成是一个族。
- 抽象工厂模式解决了工厂模式的一个问题,即工厂模式中如果有大量的子类对象,那么就会有大量的具体工厂类。但是抽象工厂模式也有一个缺点,就是在抽象工厂模式设计最初,就要考虑好产品的等级结构,例如地图分为院内地图和院外地图,保证以后不会再增加院X地图。因为抽象工厂模式,对于增加一个产品族,如加入高德的院内院外地图可以不修改原有代码,遵循了开闭原则,但是对于增加等级结构或者说是修改等级结构,那么会需要修改抽象工厂类的代码,这样会导致底下所有的工厂类都需要修改。这也被称作是抽象工厂模式的开闭原则的倾斜性。即支持增加新的产品族但是不支持增加新的产品结构。