工厂模式有简单工厂模式,工厂方法模式和抽象工厂模式几种形态。其中简单工厂模式和工厂方法模式已经在前面作过介绍。在这里,我们来介绍抽象工厂模式。
抽象工厂模式是所有形态的工厂模式中最为抽象和最具广泛性的一种形态。
抽象工厂模式的定义
抽象工厂模式是工厂方法模式的进一步扩广化和抽象化。我们给出抽象工厂模式的类图定义如下。
图1. 抽象工厂模式的类图定义
从上图可以看出,简单工厂模式涉及到以下的角色
抽象工厂(AbstractFactory)类或接口
担任这个角色的是工厂方法模式的核心,它是与应用程序无关的。任何在模式中创立对象的工厂类必须实现这个接口,或继承这个类。
实工厂类 (Conrete Factory)
担任这个角色的是与应用程序紧密相关的,直接在应用程序调用下,创立产品实例的那样一些类。
抽象产品 (Abstract Product)
担任这个角色的类是工厂方法模式所创立的对象的父类,或它们共同拥有的接口。
实产品 (Concrete Product)
担任这个角色的类是工厂方法模式所创立的任何对象所属的类。
怎么这个类图和工厂方法模式的类图看起来是一样的?
是的,图是一样的,但是含义有很大的不同。必须指出,在抽象工厂模式中,抽象产品 (AbstractProduct) 可能是一个或多个,从而构成一个或多个产品族(Product Family)。 在只有一个产品族的情况下,抽象工厂模式实际上退化到工厂方法模式。在上面的类图中,只给出了一个产品族,相当于位图中的一个点,而完整的位图应当是三维的,如下图。
图2. 抽象工厂模式的位图
从位图可以清楚地看到,与纸面垂直的数轴,即第三维轴,是代表产品族的数轴。上面的位图中展示的是有两个产品族,族A和族B的情形。
在只有一个产品族时,第三维就坍缩掉,位图也就只剩下两维。这时抽象工厂模式就退化得与工厂方法模式一模一样。
在什么情形下应当使用抽象工厂模式
在以下情况下,应当考虑使用抽象工厂模式。
首先,一个系统应当不依赖于产品类实例被创立,组成,和表示的细节。这对于所有形态的工厂模式都是重要的。
其次,这个系统的产品有多于一个的产品族。
第三,同属于同一个产品族的产品是设计成在一起使用的。这一约束必须得在系统的设计中体现出来。
最后,不同的产品以一系列的接口的面貌出现,从而使系统不依赖于接口实现的细节。
其中第二丶第三个条件是我们选用抽象工厂模式而非其它形态的工厂模式的关键性条件。
抽象工厂模式在小花果园系统中的实现
现在,我们在佛罗里达的渡假小屋修整好啦。接下来,一项重要而光荣的工作,就是开发小屋后面的小花园。这意味着,我们有两处小花园需要照料,一处在北方地区,另一处在亚热带地区。抽象工厂模式正好适用于我们的情况。
图3. 抽象工厂模式应用于小花果园系统中。三种不同的背景颜色可以区分工厂类,蔬菜类(第一产品族),和水果类的类图(第二产品族)
两处花园就相当于两个产品族。显然,给北方花园的植物是要种植在一起的,给南方花园的植物是要另种植在一起的。这种分别应当体现在系统的设计上面。这就满足了应当使用抽象工厂模式的第二和第三个条件。
package com.javapatterns.abstractfactory;
public interface Gardener {}
代码清单1. 接口 Gardener。
package com.javapatterns.abstractfactory;
public class NorthenGardener implements Gardener
{
public VeggieIF createVeggie(String name) { return new NorthernVeggie(name); }
public FruitIF createFruit(String name) { return new NorthernFruit(name); }
}
代码清单2. 实工厂类 NorthenGardener。
package com.javapatterns.abstractfactory;
public class TropicalGardener implements Gardener
{
public VeggieIF createVeggie(String name) { return new TropicalVeggie(name); }
public FruitIF createFruit(String name) { return new TopicalFruit(name); }
}
代码清单3. 实工厂类 TropicalGardener。
package com.javapatterns.abstractfactory;
public interface VeggieIF {}
代码清单4. 接口 VeggieIF。
package com.javapatterns.abstractfactory;
public class NorthernVeggie implements VeggieIF
{
public NorthernVeggie(String name) { this.name = name; }
public String getName(){ return name; }
public void setName(String name){ this.name = name; }
private String name;
}
代码清单5. 实产品类 NorthernVeggie。实产品类 NorthernFruit 与此极为类似,故略去。
package com.javapatterns.abstractfactory;
public class TropicalVeggie implements VeggieIF
{
public TropicalVeggie(String name) { this.name = name;}
public String getName(){ return name; }
public void setName(String name){ this.name = name; }
private String name;
}
代码清单6. 实产品类 TropicalVeggie。实产品类 TropicalFruit 与此极为类似,故略去。
笔者对植物的了解有限,为免遗笑大方,在上面的系统里采用了简化处理。没有给出高纬度和低纬度的水果类或蔬菜类的具体名称。
抽象工厂模式的另一个例子
这个例子讲的是微型计算机的生产。产品族有两个,PC(IBM系列)和Mac(MacIntosh系列)。显然,我们应该使用抽象工厂模式,而不是工厂方法模式,因为后者适合于处理只有一个产品族的情形。
图4. 抽象工厂模式应用于微型计算机生产系统中。两种不同的背景颜色可以区分两类产品族,及其对应的实工厂类
关于模式的实现
在抽象实现工厂模式时,有下面一些值得注意的技巧。
第一丶实工厂类可以设计成单态类。很显然,在小花果园系统中,我们只需要 NorthenGardener 和TropicalGardener 的一个实例就可以了。关于单态类的知识,请见<爪哇语言单态类创立性模式>。
第二丶在实现抽象工厂模式时,产品类往往分属多于一个的产品族,而针对每一族,都需要一个实工厂类。在很多情况下,几个实工厂类都彼此相象,只有些微的差别。
这时,笔者建议使用原始模型(Prototype)模式。这一模式会在以后介绍,届时作者会进一步阐述这一点。
第三丶设计更加灵活的实工厂。以微型计算机生产系统为例,PCProducer 是一个实工厂类,它的不灵活之处在于,每一种产品都有一个工厂方法。CPU 有createCPU(),RAM 有createRAM(),等等。如果一个已有的系统需要扩充,比如增加硬盘这一新产品,我们就需要增加一系列的接口 (createHD())丶类(HD, PCHD, MacHD)和方法。这似乎不很理想。
一个解决的办法是,把createCPU(),createRAM(), createHD()这几个方法合并为一个createPart(String type)方法。这个合并后的方法返还一个Part接口。所有的产品都要实现这一接口,而CPU,RAM,和HD接口则不再需要了。每一个实产品都需要有一个属性,表明它们的种类是CPU,RAM,和HD。
这样做的结果是,数据类型的丰富结构被扁平化了。客户端拿到的永远是一个Part接口。这对客户端而言不很安全。
第四丶抽象工厂类可以配备静态方法,以返还实工厂。设计的方法有两种。
一种是以一个静态方法,按照参量的值,返回所对应的实工厂。静态方法的数据类型是抽象方法类。
另一种是以每一个实工厂类都配备一个静态方法,其数据类型是该实工厂类。
问答题
第1题。如上面的讨论,抽象工厂类可以配备一个静态方法,按照参量的值,返回所对应的实工厂。请把微型计算机生产系统的抽象工厂类按照这一方案改造,给出UML类图和源代码。
第2题。如上面的讨论,抽象工厂类可以配备一系列静态方法对应一系列的实工厂。请把微型计算机生产系统的抽象工厂类按照这一方案改造,给出UML类图和源代码。
第3题。如上面的讨论,实工厂类可以设计成单态类。请在第1题的基础上把微型计算机生产系统的实工厂类按照这一方案改造,给出UML类图和源代码。
问答题答案
第1题。微型计算机生产系统的抽象工厂原本是接口,现在需要改造成抽象类。
图5. 三种不同的背景颜色可以区分抽象工厂类,两类产品族,及其对应的实工厂类。ComputerProducer 类图中类名为斜体表明该类是抽象的,而getProducer()的下划线表明该方法是静态的
package com.javapatterns.abstractfactory.exercise1;
public class ComputerProducer
{
public static ComputerProducer getProducer(String which)
{
if (which.equalsIgnoreCase("PC"))
{
return new PCProducer();
}
else (which.equalsIgnoreCase("Mac"))
{
return new MacProducer();
}
}
}
代码清单7. 抽象类 ComputerProducer 的方法 getProducer(String which)。
第2题。略。
第3题。本题答案是在第1题基础之上的。