〇、小故事
在《小王的披萨店》这篇文章中,我们介绍了小王开披萨店的故事,并且为了解决多种口味披萨的扩展问题,引出了简单工厂和工厂方法模式。但是,故事仍在继续,如果芝加哥的披萨店和纽约的披萨店,要求往披萨上面加的配料都不一样,那么如何可以规范多种类型披萨的创建呢?如下图所示:
【Dough】生面团
【Sauce】调味汁
【Cheese】干酪、奶酪
【Clams】蛤蜊
那么为了解决类似的问题,就可以采用我们今天要介绍的模式——抽象工厂模式。它解决了的是同一类型下多个产品族的创建。通过抽象工厂,组合了多种类型产品的创建(类似产品线)。
一、模式定义
抽象工厂模式(Abstract Factory Pattern
)
提供一个接口,用于创建相关或依赖对象的家族,而不需要明确指定具体类。
二、模式类图
对于抽象工厂,将一系列产品(配料类型,比如:Dough
、Sauce
、Cheese
、Clams
)统一到一起去创建(即:PizzaIngredientFactory的实现类:NYPizzaingredientFactory
和ChicagoPizzaingredientFactory
)。PizzaIngredientFactory中的createDough()
、createSauce()
等方法,不关心底层配料如何实现,它只关系产品类型,不在意创建细节。具体类图,如下所示:
三、代码实现
奶酪原料及实现类:Cheese.java
、MozzarellaCheese.java
、ReggianoCheese.java
/** 奶酪接口 **/
public interface Cheese {
void description();
}
/** 莫泽雷勒干酪 **/
public class MozzarellaCheese implements Cheese {
public void description() {
System.out.println("MozzarellaCheese");
}
}
/** 帕尔玛奶酪 **/
public class ReggianoCheese implements Cheese {
public void description() {
System.out.println("ReggianoCheese");
}
}
蛤蜊原料及实现类:Clams.java
、FreshClams.java
、FrozenClams.java
/** 蛤蜊接口 **/
public interface Clams {
void description();
}
/** 新鲜蛤蜊 **/
public class FreshClams implements Clams{
public void description() {
System.out.println("FreshClams");
}
}
/** 冷冻蛤蜊 **/
public class FrozenClams implements Clams{
public void description() {
System.out.println("FrozenClams");
}
}
生面团原料及实现类:Dough.java
、ThickCrustDough.java
、ThinCrustDough.java
/** 生面团接口 **/
public interface Dough {
void description();
}
/** 厚的面包皮生面团 **/
public class ThickCrustDough implements Dough {
public void description() {
System.out.println("ThickCrustDough");
}
}
/** 薄的面包皮生面团 **/
public class ThinCrustDough implements Dough {
public void description() {
System.out.println("ThinCrustDough");
}
}
调味汁原料及实现类:Sauce.java
、MarinaraSauce.java
、PlumTomatoSauce.java
/** 调味汁接口 **/
public interface Sauce {
void description();
}
/** 番茄酱调味汁 **/
public class MarinaraSauce implements Sauce {
public void description() {
System.out.println("MarinaraSauce");
}
}
/** 梅子西红柿调味汁 **/
public class PlumTomatoSauce implements Sauce {
public void description() {
System.out.println("PlumTomatoSauce");
}
}
披萨原料抽象工厂及实现类:PizzaIngredientFactory.java
、NYPizzaingredientFactory.java
、ChicagoPizzaingredientFactory.java
/** 披萨原料抽象工厂 **/
public interface PizzaIngredientFactory {
Dough createDough();
Sauce createSauce();
Cheese createCheese();
Clams createClams();
}
/** 芝加哥原料工厂 **/
public class ChicagoPizzaingredientFactory implements PizzaIngredientFactory {
public Dough createDough() {
return new ThinCrustDough(); // 薄的面包皮生面团
}
public Sauce createSauce() {
return new MarinaraSauce(); // 番茄酱调味汁
}
public Cheese createCheese() {
return new ReggianoCheese(); // 帕尔玛奶酪
}
public Clams createClams() {
return new FreshClams(); // 新鲜蛤蜊
}
}
/** 纽约原料工厂 **/
public class NYPizzaingredientFactory implements PizzaIngredientFactory {
public Dough createDough() {
return new ThickCrustDough(); // 厚的面包皮生面团
}
public Sauce createSauce() {
return new PlumTomatoSauce(); // 梅子西红柿调味汁
}
public Cheese createCheese() {
return new MozzarellaCheese(); // 莫泽雷勒干酪
}
public Clams createClams() {
return new FrozenClams(); // 冷冻蛤蜊
}
}
创建Pizza及实现类:Pizza.java
/** 披萨抽象类 **/
public abstract class Pizza {
protected String name;
protected Dough dough; // 面团
protected Sauce sauce; // 酱
protected Cheese cheese; // 干酪
protected Clams clams; // 蛤蜊
protected PizzaIngredientFactory pizzaIngredientFactory;
/** 准备原材料 */
public abstract void prepare();
public void bake() {
System.out.println("pizzaIngredientFactory bake()");
}
public void cut() {
System.out.println("pizzaIngredientFactory cut()");
}
public void box() {
System.out.println("pizzaIngredientFactory box()");
}
}
奶酪口味披萨:CheesePizza.java
/** 奶酪口味披萨 **/
public class CheesePizza extends Pizza {
public CheesePizza(PizzaIngredientFactory pizzaIngredientFactory) {
this.pizzaIngredientFactory = pizzaIngredientFactory;
}
public void prepare() {
System.out.println("CheesePizza prepare()");
dough = pizzaIngredientFactory.createDough();
sauce = pizzaIngredientFactory.createSauce();
cheese = pizzaIngredientFactory.createCheese();
}
}
蛤蜊口味披萨:ClamPizza.java
/** 蛤蜊口味披萨 **/
public class ClamPizza extends Pizza {
public ClamPizza(PizzaIngredientFactory pizzaIngredientFactory) {
this.pizzaIngredientFactory = pizzaIngredientFactory;
}
public void prepare() {
System.out.println("ClamPizza prepare()");
dough = pizzaIngredientFactory.createDough();
sauce = pizzaIngredientFactory.createSauce();
cheese = pizzaIngredientFactory.createCheese();
// 加入蛤蜊配料
clams = pizzaIngredientFactory.createClams();
}
}
意大利辣肉肠口味披萨:PepperoniPizza.java
/**意大利辣肉肠口味披萨 **/
public class PepperoniPizza extends Pizza {
public PepperoniPizza(PizzaIngredientFactory pizzaIngredientFactory) {
this.pizzaIngredientFactory = pizzaIngredientFactory;
}
public void prepare() {
System.out.println("PepperoniPizza prepare()");
dough = pizzaIngredientFactory.createDough();
sauce = pizzaIngredientFactory.createSauce();
cheese = pizzaIngredientFactory.createCheese();
// 加入蛤蜊配料
clams = pizzaIngredientFactory.createClams();
}
}
蔬菜口味披萨:VeggiePizza.java
/** 蔬菜口味披萨 **/
public class VeggiePizza extends Pizza {
public VeggiePizza(PizzaIngredientFactory pizzaIngredientFactory) {
this.pizzaIngredientFactory = pizzaIngredientFactory;
}
public void prepare() {
System.out.println("VeggiePizza prepare()");
dough = pizzaIngredientFactory.createDough();
sauce = pizzaIngredientFactory.createSauce();
cheese = pizzaIngredientFactory.createCheese();
}
}
披萨工厂及实现类:PizzaStoreV3.java
、NYPizzaStore.java
public abstract class PizzaStoreV3 {
protected abstract Pizza createPizza(String pizzaType);
public Pizza orderPizza(String pizzaType) {
Pizza pizza = createPizza(pizzaType);
pizza.prepare();
pizza.bake();
pizza.cut();
pizza.box();
return pizza;
}
}
public class NYPizzaStore extends PizzaStoreV3 {
protected Pizza createPizza(String pizzaType) {
Pizza pizza = null;
PizzaIngredientFactory pizzaIngredientFactory = new NYPizzaingredientFactory();
if (pizzaType.equals("cheese")) {
pizza = new CheesePizza(pizzaIngredientFactory);
} else if (pizzaType.equals("pepperoni")) {
pizza = new PepperoniPizza(pizzaIngredientFactory);
} else if (pizzaType.equals("clam")) {
pizza = new ClamPizza(pizzaIngredientFactory);
} else if (pizzaType.equals("veggie")) {
pizza = new VeggiePizza(pizzaIngredientFactory);
}
return pizza;
}
}
抽象工厂测试类:PizzaStoreTest.java
public class PizzaStoreTest {
public static void main(String[] args) {
PizzaStoreV3 pizzaStore = new NYPizzaStore();
pizzaStore.orderPizza("cheese");
}
}
四、工厂方法和抽象工厂的区别
抽象工厂的方法经常以工厂方法的方式实现。也就是说,工厂方法经常会潜伏在抽象工厂里面。
工厂方法的特点:
【1】采用继承的方法实现。
【2】通过抽象方法,来通过子类实现该方法,生成对象。
抽象工厂的特点:
【1】采用组合的方法实现。
【2】通过提供接口,来创建一组产品的接口。这个接口内的每个方法都负责创建一个具体产品。
今天的文章内容就这些了:
写作不易,笔者几个小时甚至数天完成的一篇文章,只愿换来您几秒钟的 点赞 & 分享 。
更多技术干货,欢迎大家关注公众号“爪哇缪斯” ~ \(^o^)/ ~ 「干货分享,每天更新」