在了解抽象工厂模式之前,首先来了解一下两个术语:
产品等级结构也就是产品的继承结构,例如一个抽象类是电视机,子类有不同品牌的电视机,比如海尔电视机,海信电视机,TCL电视机,而抽象电视机与具体品牌的电视机之间构成了一个产品等级结构,抽象电视机是父类,而具体品牌的电视机是子类。
产品族是指由一个同一个工厂产生的位于不同产品等级结构中的一组产品,例如海尔电器工厂生产的海尔电视机,海尔电冰箱。海尔电视机位于电视机产品等级结构中,海尔电冰箱位于电冰箱产品等级结构中,海尔电视机与海尔电冰箱共同构成了一个产品族。
两者示意图如下:
抽象工厂模式与工厂方法模式最大的区别在于,工厂方法模式是针对一个产品等级结构,而抽象工厂模式需要面对多个产品等级结构,一个工厂等级结构可以负责多个不同产品等级结构中的产品对象的创建。
每一个具体工厂可以生产属于一个产品族的所有产品,所生产的产品又位于不同的产品等级结构中,在上图的例子中,如果使用工厂方法模式需要12个具体工厂类,而使用抽象工厂模式只需要4个工厂类,大大减少了系统中类的个数。
抽象工厂模式:提供一个创建一系列相关或相互依赖对象的接口,而无须指定它们具体的类。
抽象工厂模式又叫Kit模式,是一种对象创建型模式。
AbstractFactory
(抽象工厂):声明了一组用于创建一族产品的方法,每一个方法对应一种产品ConcreteFactory
(具体工厂):实现了在抽象工厂中声明的创建产品的方法,生成一组具体产品,这些产品构成了一个产品族,每一个产品都位于某个产品的等级结构中AbstractProduct
(抽象产品):为每种产品声明接口,在抽象产品中声明了产品所具有的业务方法ConcreteProduct
(具体产品):定义具体工厂生产的具体对象,实现在抽象层产品接口中声明的业务方法这里简单定义两个抽象产品接口,也就是假设有两个产品等级结构(A与B):
interface ProductA
{
void methodA();
}
interface ProductB
{
void methodB();
}
声明四个具体产品,假设每种产品由分别两个工厂生产,也就是假设有两个产品族:
class ProductA1 implements ProductA
{
public void methodA()
{
System.out.println("Product A1");
}
}
class ProductA2 implements ProductA
{
public void methodA()
{
System.out.println("Product A2");
}
}
class ProductB1 implements ProductB
{
public void methodB()
{
System.out.println("Product B1");
}
}
class ProductB2 implements ProductB
{
public void methodB()
{
System.out.println("Product B2");
}
}
由于只有两个产品等级结构(A与B),这里的抽象工厂只需要两个方法,分别表示生产这两种产品:
interface Factory
{
ProductA getProductA();
ProductB getProductB();
}
由于上面已经假设为两个产品族,因此这里需要两个具体工厂:
//产品族Factory1
class Factory1 implements Factory
{
public ProductA getProductA()
{
return new ProductA1();
}
public ProductB getProductB()
{
return new ProductB1();
}
}
//产品族Factory2
class Factory2 implements Factory
{
public ProductA getProductA()
{
return new ProductA2();
}
public ProductB getProductB()
{
return new ProductB2();
}
}
public static void main(String[] args) {
Factory factory = new Factory1();
ProductA productA = factory.getProductA();
ProductB productB = factory.getProductB();
productA.methodA();
productB.methodB();
factory = new Factory2();
productA = factory.getProductA();
productB = factory.getProductB();
productA.methodA();
productB.methodB();
}
客户端针对抽象工厂以及抽象产品编程,只需要知道工厂类名即可获取同一工厂(同一产品族)不同产品等级结构的产品。
界面皮肤库设计:开发一套皮肤库,用户可以通过菜单选择皮肤,不同的皮肤提供视觉不同的按钮,文本框等UI元素,使用抽象工厂模式进行设计。
这里简单起见假设开发两套皮肤:
每套皮肤具有以下UI元素:
具体设计如下:
Button
+TextField
+ComboBox
SpringButton
+SummerButton
+SpringTextField
+SummerTextField
+SpringComboBox
+SummerComboBox
SkinFactory
SpringSkinFactory
+SummerSkinFactory
先设计产品类:
//抽象产品
interface Button
{
void display();
}
//具体产品
class SpringButton implements Button
{
public void display()
{
System.out.println("春季皮肤按钮");
}
}
//具体产品
class SummerButton implements Button
{
public void display()
{
System.out.println("夏季皮肤按钮");
}
}
//抽象产品
interface TextField
{
void display();
}
//具体产品
class SpringTextField implements TextField
{
public void display()
{
System.out.println("春季皮肤文本框");
}
}
//具体产品
class SummerTextField implements TextField
{
public void display()
{
System.out.println("夏季皮肤文本框");
}
}
//抽象产品
interface ComboBox
{
void display();
}
//具体产品
class SpringComboBox implements ComboBox
{
public void display()
{
System.out.println("春季皮肤组合框");
}
}
//具体产品
class SummerComboBox implements ComboBox
{
public void display()
{
System.out.println("夏季皮肤组合框");
}
}
接着是工厂类:
//抽象工厂
interface SkinFactory
{
Button createButton();
TextField createTextField();
ComboBox createComboBox();
}
//具体工厂
class SpringSkinFactory implements SkinFactory
{
public Button createButton()
{
return new SpringButton();
}
public TextField createTextField()
{
return new SpringTextField();
}
public ComboBox createComboBox()
{
return new SpringComboBox();
}
}
//具体工厂
class SummerSkinFactory implements SkinFactory
{
public Button createButton()
{
return new SummerButton();
}
public TextField createTextField()
{
return new SummerTextField();
}
public ComboBox createComboBox()
{
return new SummerComboBox();
}
}
测试:
public class Test
{
public static void main(String[] args) {
SkinFactory factory = new SpringSkinFactory();
factory.createButton().display();
factory.createTextField().display();
factory.createComboBox().display();
factory = new SummerSkinFactory();
factory.createButton().display();
factory.createTextField().display();
factory.createComboBox().display();
}
}
虽然使用抽象工厂模式增加新的皮肤界面非常方便,但是如果增加一个UI元素,会修改大量的代码,需要修改抽象工厂以及每一个具体工厂类,也就是说,不能够在符合OCP(开放闭合原则)的前提下增加新的组件。
这是抽象工厂模式的最大缺点,尽管增加新的产品族(这里是皮肤)非常方便,但是增加新的产品等级结构(这里是UI元素)很麻烦。抽象工厂模式的这种性质叫做开闭原则的倾斜性。因此设计人员在设计之初需要全面考虑,否则新增产品结构会导致大量的代码修改。
主要缺点是增加新的产品等级结构麻烦,需要对系统进行大量的修改,违背了OCP。
如果觉得文章好看,欢迎点赞。
同时欢迎关注微信公众号:氷泠之路。