Head First设计模式(阅读笔记)-04.工厂模式
创始人
2024-04-12 21:27:35
0

披萨订购

假设要完成披萨订购的功能,披萨的种类很多,比如 GreekPizzCheesePizz 等,披萨店会根据用户需要的披萨种类制作披萨,制作的流程包括prepare->bake->cut->box


简单实现

下面代码的实现十分简单清晰,但是如果披萨类型增加或删除,需要去修改代码,不符合开闭原则(类应该对扩展开发,对修改关闭)


// 下面不同的pizza类型都继承了相同的父类
Pizza orderPizza(String type){Pizza pizza;if(type.equals("cheese")){pizza = new CheesePizza();}else if(type.equals("clam")){pizza = new ClamPizza();}pizza.prepare();pizza.bake();pizza.cut();pizza.box();return pizza;
}

工厂模式


简单的披萨工厂-简单工厂模式

从上述代码可以看出,出了创建披萨对象的代码需要修改,其他代码都是不变的,所以可以将创建披萨对象的代码分离出来作为一个新对象-披萨工厂:

  • 此时orderPizza方法变为该工厂的客户,它无需知道工厂如何得到一个披萨,只关心从工厂可以得到一个披萨,并且可以调用方法对披萨进行准备、烘烤等操作
  • 并且该工厂可以有许多客户,比如制作披萨点的菜单就可以利用该工厂制作的披萨来制定价钱等

// 简单披萨工厂
public class SimplePizzaFactory{public Pizza createPizza(String type){Pizza pizza = null;if(type.equals("cheese")){pizza = new CheesePizza();}else if(type.equals("clam")){pizza = new ClamPizza();}return pizza;}
}
public class PizzaStore{SimplePizzaFactory factory;// 披萨店需要指定一个工厂public PizzaStore(SimplePizzaFactory factory){this.factory = factory;}public Pizza orderPizza(String type){Pizza pizza;factory.createPizza(type);pizza.prepare();pizza.bake();pizza.cut();pizza.box();return pizza;
}

加盟披萨店-工厂方法模式

此时在纽约等地区想在当地加盟该披萨店,并且希望工厂能加工出纽约风味的披萨


给每个地区创建一个工厂

下面代码的缺陷在于无法让各地区的加盟店决定自己的制作流程,只是决定采用哪个工厂而已

NYPizzaFactory nyFactory = new NYPizzaFactory();
PizzaStore nyStore = new PizzaStore(nyFactory);
nyStore.orderPizza("Veggie");  // 在纽约工厂制作的纽约风味披萨
让加盟店自己决定制作流程

原本由一个对象负责所有具体类的实例化,现在通过子类负责实例化


public abstract class PizzaStore{// orderPizza可以声明为final,防止被子类覆盖public Pizza orderPizza(String type){Pizza pizza;pizza = createPizza(type)  // 从createPizza方法从工厂中移到披萨店中pizza.prepare();pizza.bake();pizza.cut();pizza.box();return pizza;}// 定义为抽象,必须实现protected abstract Pizza createPizza(String type);
}public class NYPizzaStrore extends PizzaStore{// 子类全权负责实例化哪个具体Pizza,父类并不知道具体创建的披萨是哪一种,只知道这个披萨要进行烘烤、切片等流程Pizza createPizza(String item){if(item.equals("chess")){return new NYStyleCheesePizza();}else if(...){...}}
}// 定义一个抽象的披萨类
public abstract class Pizza{String name;void cut(){System.out.println("切成方块");}
}// 纽约风味的披萨
public class NYStyleCheesePizza extends Pizza{public NYStyleCheesePizza(){name = "NYStyle with cheese"}void cut(){System.out.println("切成三角形");}
}// 测试
public class Test{public static void main(String[] agrs){PizzaStore nyStore = new NYPizzaStrore();Pizza pizza = nyStore.orderPizza("cheese");  // nyStore决定要制作什么披萨}
}
工厂方法模式

定义了一个创建对象的接口,由子类决定要实例化哪个类


在这里插入图片描述

依赖倒置原则

对于工厂方法模式还依赖了一个原则-依赖倒置原则,该原则表示要依赖抽象,不要依赖具体类,换个说法就是高层组件和低层组件都应该依赖于抽象


在这里插入图片描述

使用当地原材料-抽象工厂模式

假设不同地区的加盟店制作披萨要求用当地的原材料制作


创建当前原料工厂

很明显可以想到,不可能只存在一家工厂生产不同地区需要的原料,然后空运过去,可以在当地创建一个原料工厂


// 抽象原料工厂
public interface PizzaIngredientFactory{public Sauce createSauce();public Cheese createCheese();
}
// 纽约的原料工厂
public class NYPizzaIngredientFactory implements PizzaIngredientFactory{public Sauce createSauce(){return new NYSauce();  // 使用当前材料}...
}
重新制作披萨

// 抽象披萨类,将prepare方法定义为抽象了
public abstract class Pizza{String name;Sauce sauce;Cheese Cheese;// 声明为抽象是因为现在当地披萨需要使用自己的原材料进行准备abstract void prepare();void cut(){System.out.println("切成方块");}
}
// 现在根据使用不同的原料工厂就可以区分出是哪个地区的CheesePizza了
public class CheesePizza extends Pizza{PizzaIngredientFactory ingredientFactory;// 只需要传入需要地区的原料工厂即可public CheesePizza(PizzaIngredientFactory ingredientFactory){this.ingredientFactory = ingredientFactory;}void prepare(){// 通过传入的原料工厂就可以生产出该工程的saucesauce = ingredientFactory.createSauce();...}
}public class NYPizzaStore extends PizzaStore{Pizza createPizza(String item){Pizza pizza = null;// 纽约加盟店肯定使用纽约原料工厂了PizzaIngredientFactory ingredientFactory = new NYPizzaIngredientFactory();if(item.equals("chess")){// 现在不需要写为NYStyleCheesePizza,只需要传入对于原料工厂即可pizza = new CheesePizza(ingredientFactory);}else if(...){...}return pizza;}
}// 测试(代码没有变化)
public class Test{public static void main(String[] agrs){PizzaStore nyStore = new NYPizzaStrore();Pizza pizza = nyStore.orderPizza("cheese");}
}
抽象工厂模式

提供一个接口用于创建相关或依赖对象的家族,并且不需要明确具体类(比如在NYPizzaStorecreatePizza方法中,并没有明确具体的NYStyleCheesePizza,而是传入一个原料工厂)


参考

Head First 设计模式-工厂模式

相关内容

热门资讯

个人销售住房增值税政策来了 12月29日,财政部、国家税务总局发布“关于个人销售住房增值税政策的公告”。 个人(不含个体工商...
南京拟修订法规加强历史文化名城... 原题:保护利用将设“正负面清单” 《南京市历史文化名城保护条例》将迎大修 南京市历史文化名城保护法规...
《深圳公益诉讼检察工作白皮书(... 深圳新闻网2025年12月31日讯(深圳特区报记者 上官文复 通讯员 李俞青)12月30日,深圳市人...
了解招生政策要选择官方渠道 石向阳 绘 日前,清华大学招生办公室发布声明称,有部分机构与个人冒用学校名义开展招生宣讲,散布不实招...
视频丨日学者:高市言行和政策令... 日本上智大学政治学教授中野晃一近日在接受总台记者采访时指出,高市早苗有关对华立场的言论将进一步破坏日...
10年受理公益诉讼案件线索超8... 深圳特区报讯(记者 上官文复 通讯员 李俞青)12月30日,深圳市人民检察院举办《深圳公益诉讼检察工...
日学者:高市言行和政策令日本面... 日本上智大学政治学教授中野晃一近日在接受总台记者采访时指出,高市早苗有关对华立场的言论将进一步破坏日...
袁家军、胡衡华、刘明胜、徐树彪... 据重庆日报消息,12月30日下午, 国家电投集团水电股份有限公司揭牌。重庆市委书记袁家军,市委副书记...
“3女带4孩续面”被改编成动画... 极目新闻记者 詹钘 近日,有网友发现,郑州续面事件已经被人改编成动画,在短视频平台和短剧平台播放。视...
征兵政策解读之二:征集条件篇 一、年龄条件 1.义务兵。男青年年满18 至22 周岁,普通高等学校本专科毕业生、符合毕业条件的毕业...