《Head first设计模式》之工厂模式
工厂方法模式定义了一个创建对象的接口,但由子类决定要实例化的类是哪一个。工厂方法让类把实例化推迟到了子类。
预定披萨
假设你有一个披萨店,预定披萨的代码可能是这么写的:
Pizza orderPizza(){
Pizza pizza = new Pizza();
// 准备面皮,加调料等
pizza.prepare();
// 烘烤
pizza.bake();
// 切片
pizza.cut();
// 装盒
pizza.box();
return pizza;
}
更多的披萨类型
但是,你现在需要更多的披萨类型。你必须增加一些代码,来“决定”适合的披萨类型,然后再“制造”这个披萨:
// 现在把披萨类型传入orderPizza
Pizza orderPizza(String type) {
Pizza pizza;
// 根据披萨的类型,我们实例化正确的具体类,然后将其赋值给pizza实例变量。
// 请注意,这里的任何披萨都必须实现pizza接口。
if (type.equals("cheese")) {
pizza = new CheesePizza();
} else if (type.equals("greek")) {
pizza = new GreekPizza();
} else if (type.equals("pepperoni")) {
pizza = new PepperoniPizza();
}
pizza.prepare();
pizza.bake();
pizza.cut();
pizza.box();
return pizza;
}
修改披萨类型
你发现你所有的竞争者都已经在他们的菜单中加入了一些流行风味的披萨:Clam Pizza(蛤蜊披萨)、Veggie Pizza(素食披萨)。很明显,你必须要赶上他们,所以也要把这些风味加进你的菜单中。而最近Greek Pizza(希腊披萨)卖得不好,所以你决定将它从菜单中去掉:
Pizza orderPizza(String type) {
Pizza pizza;
// 此代码没有对修改封闭。如果披萨店改变它所供应的披萨风味,就得进到这里进行修改。
if (type.equals("cheese")) {
pizza = new CheesePizza();
// } else if (type.equals("greek")) {
// pizza = new GreekPizza();
} else if (type.equals("pepperoni")) {
pizza = new PepperoniPizza();
} else if (type.equals("clam")) {
pizza = new ClamPizza();
} else if (type.equals("veggie")) {
pizza = new VeggiePizza();
}
// 这里是我们不想改变的地方。因为披萨的准备、烘烤、包装,多年来持续不变,
// 所以这部分的代码不会改变,只有发生这些动作的披萨会改变。
pizza.prepare();
pizza.bake();
pizza.cut();
pizza.box();
return pizza;
}
很明显地,如果实例化“某些”具体类,将使orderPizza()出问题,而且也无法让orderPizza()对修改关闭。但是,现在我们已经知道哪些会改变,哪些不会改变,该是使用封装的时候了。
建立一个简单披萨工厂
现在最好将创建对象移到orderPizza()之外,但怎么做呢?我们可以把创建披萨的代码移到另一个对象中,由这个新对象专职创建披萨。
我们称这个新对象为“工厂”。
工厂(factory)处理创建对象的细节。一旦有了SimplePizzaFactory,orderPizza()就变成此对象的客户。当需要披萨时,就叫披萨工厂做一个。那些orderPizza()方法需要知道希腊披萨或者蛤蜊披萨的日子一去不复返了。现在orderPizza()方法只关心从工厂得到了一个披萨,而这个披萨实现了Pizza接口,所以它可以调用prepare()、bake()、cut()、box()来分别进行准备、烘烤、切片、装盒。
// SimplePizzaFactory是我们的新类,它只做一件事情:帮它的客户创建披萨
public class SimplePizzaFactory {
// 首先,在这个工厂内定义一个createPizza()方法,所有客户用这个方法来实例化新对象。
public Pizza createPizza(String type) {
Pizza pizza = null;
// 这是从orderPizza()方法中移过来的代码
if (type.equals("cheese")) {
pizza = new CheesePizza();
} else if (type.equals("pepperoni")) {
pizza = new PepperoniPizza();
} else if (type.equals("clam")) {
pizza = new ClamPizza();
} else if (type.equals("veggie")) {
pizza = new VeggiePizza();
}
return pizza;
}
}
// 现在我们为PizzaStore加上一个对SimplePizzaFactory的引用
public class PizzaStore {
SimplePizzaFactory factory;
// PizzaStore的构造器,需要一个工厂作为参数
public PizzaStore(SimplePizzaFactory factory){
this.factory = factory;
}
public Pizza orderPizza(String type) {
Pizza pizza;
// orderPizza()方法通过简单传入订单类型来使用工厂创建披萨。
// 请注意,我们把new操作符替换成工厂对象的创建方法。这里不再使用具体实例化
pizza = factory.createPizza(type);
pizza.prepare();
pizza.bake();
pizza.cut();
pizza.box();
return pizza;
}
}
加盟披萨店
你的披萨店经营有成,击败了竞争者,现在大家都希望能在自家附近有加盟店。身为加盟公司经营者,你希望确保加盟店运营的质量,所以希望这些店都使用你那些经过时间考验的代码。
但是区域的差异呢?每家加盟店都可能想要提供不同风味的披萨(比方说纽约、芝加哥、加州),这受到了开店地点及该地区披萨美食家口味的影响。
在推广SimplePizzaFactory时,你发现加盟店的确是采用你的工厂创建披萨,但是其他部分,却开始采用他们自创的流程:烘烤的做法有差异、不要切片、使用其他厂商的盒子。
能不能建立一个框架,把加盟店和创建披萨捆绑在一起的同时又保持一定的弹性?
给披萨店使用的框架
有个做法可让披萨制作活动局限于PizzaStore类,而同时又能让这些加盟店依然可以自由地制作该区域的风味。
所要做的事情,就是把createPizza()方法放回到PizzaStore中,不过要把它设置成“抽象方法”,然后为每个区域风味创建一个PizzaStore的子类。
首先,看PizzaStore所做的改变:
public abstract class PizzaStore {
public Pizza orderPizza(String type) {
Pizza pizza;
// 现在createPizza()方法从工厂对象中移回PizzaStore
pizza = createPizza(type);
// 这些都没变
pizza.prepare();
pizza.bake();
pizza.cut();
pizza.box();
return pizza;
}
// 现在把工厂对象移到这个方法中
// 在PizzaStore里,“工厂方法”现在是抽象的
abstract Pizza createPizza(String type);
}
现在已经有一个PizzaStore作为超类;让每个域类型(NYPizzaStore、ChicagoPizzaStore、CaliforniaPizzaStore)都继承这个PizzaStore,每个子类各自决定如何制作披萨。
允许子类做决定
PizzaStore已经有一个不错的订单系统,由orderPizza()方法负责处理订单,而你希望所有加盟店对于订单的处理都能一致。
各个区域披萨店之间的差异在于他们制作披萨的风味(纽约披萨的薄脆、芝加哥披萨的饼厚等),我们现在要让现在createPizza()能够应对这些变化来负责创建正确种类的披萨。做法是让PizzaStore的各个子类负责定义自己的createPizza()方法。所以我们会得到一些PizzaStore具体的子类,每个子类都有自己的披萨变体,而仍然适合PizzaStore框架,并使用调试好的orderPizza()方法。
public abstract class PizzaStore {
public Pizza orderPizza(String type) {
Pizza pizza;
pizza = createPizza(type);
pizza.prepare();
pizza.bake();
pizza.cut();
pizza.box();
return pizza;
}
// 每个子类都会覆盖createPizza()方法
abstract Pizza createPizza(String type);
}
// 如果加盟店为顾客提供纽约风味的披萨,就使用NyStylePizzaStore,
// 因为此类的createPizza()方法会建立纽约风味的披萨
public class NyStylePizzaStore extends PizzaStore{
@Override
Pizza createPizza(String type) {
Pizza pizza = null;
if (type.equals("cheese")) {
pizza = new NyStyleCheesePizza();
} else if (type.equals("pepperoni")) {
pizza = new NyStylePepperoniPizza();
} else if (type.equals("clam")) {
pizza = new NyStyleClamPizza();
} else if (type.equals("veggie")) {
pizza = new NyStyleVeggiePizza();
}
return pizza;
}
}
// 类似的,利用芝加哥子类,我们得到了带芝加哥原料的createPizza()实现
public class ChicagoStylePizzaStore extends PizzaStore{
@Override
Pizza createPizza(String type) {
Pizza pizza = null;
if (type.equals("cheese")) {
pizza = new ChicagoCheesePizza();
} else if (type.equals("pepperoni")) {
pizza = new ChicagoPepperoniPizza();
} else if (type.equals("clam")) {
pizza = new ChicagoClamPizza();
} else if (type.equals("veggie")) {
pizza = new ChicagoVeggiePizza();
}
return pizza;
}
}
现在问题来了,PizzaStore的子类终究只是子类,如何能够做决定?在NyStylePizzaStore类中,并没有看到任何做决定逻辑的代码。
关于这个方面,要从PizzaStore的orderPizza()方法观点来看,此方法在抽象的PizzaStore内定义,但是只在子类中实现具体类型。
orderPizza()方法对对象做了许多事情(例如:准备、烘烤、切片、装盒),但由于Pizza对象是抽象的,orderPizza()并不知道哪些实际的具体类参与进来了。换句话说,这就是解耦(decouple)!
当orderPizza()调用createPizza()时,某个披萨店子类将负责创建披萨。做哪一种披萨呢?当然是由具体的披萨店决定。
那么,子类是实时做出这样的决定吗?不是,但从orderPizza()的角度看,如果选择在NyStylePizzaStore订购披萨,就是由这个子类(NyStylePizzaStore)决定。严格来说,并非由这个子类实际做“决定”,而是由“顾客”决定哪一家风味的披萨店才决定了披萨的风味。
工厂方法模式
工厂方法模式定义了一个创建对象的接口,但由子类决定要实例化的类是哪一个。工厂方法让类把实例化推迟到了子类。
工厂方法模式(Factory Method Pattern)通过让子类决定该创建的对象是什么,来达到将对象创建的过程封装的目的。
PizzaStore就是创建者(Creator)类。它定义了一个抽象的工厂方法,让子类实现此方法制造产品。
创建者通常会包含依赖于抽象产品的代码,而这些抽象产品由子类制造。创建者不需要真的知道在制造哪种具体产品。
能够产生产品的类称为具体创建者。NYPizzaStore和ChicagoPizzaStore就是具体创建者。
Pizza是产品类。工厂生产产品,对PizzaStore来说,产品就是Pizza。
抽象的Creator提供了一个创建对象的方法的接口,也称为“工厂方法”。在抽象的Creator中,任何其他实现的方法,都可能使用到这个工厂方法所制造出来的产品,但只有子类真正实现这个工厂方法并创建产品。
// Creator是一个类,它实现了所有操纵产品的方法,但不实现工厂方法
public abstract class Creator{
void anOperation(){
// ...
}
// Creator的所有子类都必须实现这个抽象的factoryMethod()方法
abstract void factoryMethod();
}
// 具体的创建者
public class ConcreteCreator extends Creator{
// ConcreteCreator实现了factoryMethod(),以实际制造出产品。
@Override
void factoryMethod() {
// ...
}
}
// 所有产品必须实现这个接口,这样一来,
// 使用这些产品的类就可以引用这个接口,而不是具体的类
public abstract class Product{
void operation(){
// ...
}
}
// 具体的产品
public class ConcreteProduct extends Product{
}
依赖倒置原则
假设你从未听说过OO工厂。下面是一个不使用工厂模式的披萨店版本。数一数,这个类所依赖的具体披萨对象有几种。
public class DependentPizzaStore {
public Pizza createPizza(String style, String type) {
Pizza pizza = null;
if(style.equals("NY")){
// 处理所有纽约风味的披萨
if (type.equals("cheese")) {
pizza = new NyStyleCheesePizza();
} else if (type.equals("pepperoni")) {
pizza = new NyStylePepperoniPizza();
} else if (type.equals("clam")) {
pizza = new NyStyleClamPizza();
} else if (type.equals("veggie")) {
pizza = new NyStyleVeggiePizza();
}
} else if(style.equals("Chicago")){
// 处理所有芝加哥风味的披萨
if (type.equals("cheese")) {
pizza = new ChicagoCheesePizza();
} else if (type.equals("pepperoni")) {
pizza = new ChicagoPepperoniPizza();
} else if (type.equals("clam")) {
pizza = new ChicagoClamPizza();
} else if (type.equals("veggie")) {
pizza = new ChicagoVeggiePizza();
}
} else {
System.out.println("Error");
return null;
}
pizza.prepare();
pizza.bake();
pizza.cut();
pizza.box();
return pizza;
}
}
如果把这个版本的披萨店和它依赖的对象画成一张图,看起来是这样的:
这个版本的PizzaStore依赖于所有的披萨对象,因为它直接创建这些披萨对象。
如果这些类的实现改变了,那么可能必须修改PizzaStore。
每新增一个披萨类型,就等于让PizzaStore多了一个依赖。
因为对于披萨具体实现的任何改变都会影响到PizzaStore,我们说PizzaStore“依赖于”披萨的实现。
很清楚的,代码里减少对于具体类的依赖是件好事。有一个OO设计原则就正式阐明了这一点:
依赖倒置原则:要依赖抽象,不要依赖具体类。
这个原则说明了:不能让高层组件依赖低层组件,而且,不管高层或低层组件,两者都应该依赖于抽象。
比如,这个例子里的PizzaStore是高层组件,而披萨实现是低层组件,很清楚的,PizzaStore依赖这些具体披萨类。
现在,这个原则告诉我们,应该重写代码以便于我们依赖抽象类,而不依赖具体类。对于高层及低层模块都应该如此。
依赖倒置原则的应用
非常依赖披萨店的主要问题在于:它依赖每个披萨类型。因为它是在自己的orderPizza()方法中,实例化这些具体类型的。
如何在orderPizza()方法中,将这些实例化对象的代码独立出来?我们知道,工厂方法刚好能派上用场。
所以,应用工厂方法后,类图看起来就像这样:
PizzaStore现在依赖Pizza这个抽象类。
具体披萨类也依赖Pizza抽象,因为它们实现了Pizza接口。
在应用工厂方法后,高层组件(也就是PizzaStore)和低层组件(也就是这些披萨)都依赖了Pizza抽象。想要遵循依赖倒置原则,工厂方法并非是唯一的技巧,但却是最有威力的技巧之一。
遵循依赖倒置原则的指导方针
下面的指导方针,能帮你避免在OO设计中违反依赖倒置原则:
变量不可以持有具体类的引用
如果使用new,就会持有具体类的引用。你可以改用工厂来避开这样的做法。
不要让类派生自具体类
如果派生自具体类,你就会依赖具体类。请派生自一个抽象(接口或抽象类)。
不要覆盖基类中已实现的方法
如果覆盖基类已实现的方法,那么你的基类就不是一个真正适合被继承的抽象。基类中已实现的方法,应该由所有的子类共享。
要完全遵守这些指导方针似乎不太可能,但是如果你深入体验这些方针,将这些方针内化成你思考的一部分,那么在设计时,你将知道何时有足够的理由违反这样的原则。比方说,如果有一个不像是会改变的类,那么在代码中直接实例化具体类也就没什么大碍。另一方面,如果有个类可能改变,你可以采用一些好技巧(例如工厂方法)来封装改变。
《Head first设计模式》之工厂模式的更多相关文章
- 设计模式——抽象工厂模式及java实现
设计模式--抽象工厂模式及java实现 设计模式在大型软件工程中很重要,软件工程中采用了优秀的设计模式有利于代码维护,方便日后更改和添加功能. 设计模式有很多,而且也随着时间在不断增多,其中最著名的是 ...
- 5. 星际争霸之php设计模式--抽象工厂模式
题记==============================================================================本php设计模式专辑来源于博客(jymo ...
- 3. 星际争霸之php设计模式--简单工厂模式
题记==============================================================================本php设计模式专辑来源于博客(jymo ...
- iOS 设计模式之工厂模式
iOS 设计模式之工厂模式 分类: 设计模式2014-02-10 18:05 11020人阅读 评论(2) 收藏 举报 ios设计模式 工厂模式我的理解是:他就是为了创建对象的 创建对象的时候,我们一 ...
- 设计模式之工厂模式(Factory)
设计模式的工厂模式一共有三种:简单工厂模式,工厂模式,抽象工厂模式 简单工厂模式原理:只有一个工厂类,通过传参的形式确定所创建的产品对象种类 代码如下: #include <stdio.h> ...
- php设计模式:工厂模式
php设计模式:工厂模式 意图: 定义一个用于创建对象的接口,让子类决定实例化哪一个类. 工厂模式实现: 工厂模式中任何创建对象的工厂类都要实现这个接口,实现接口的方法体中都要实现接口中的方法,它声明 ...
- 浅析JAVA设计模式之工厂模式(一)
1 工厂模式简单介绍 工厂模式的定义:简单地说,用来实例化对象,取代new操作. 工厂模式专门负责将大量有共同接口的类实例化.工作模式能够动态决定将哪一个类实例化.不用先知道每次要实例化哪一个类. 工 ...
- java 设计模式之工厂模式与反射的结合
工厂模式: /** * @author Rollen-Holt 设计模式之 工厂模式 */ interface fruit{ public abstract void eat(); } ...
- C#学习之设计模式:工厂模式
最近研究一下设计模式中工厂模式的应用,在此记录如下: 什么是工厂模式? 工厂模式属于设计模式中的创造型设计模式的一种.它的主要作用是协助我们创建对象,为创建对象提供最佳的方式.减少代码中的耦合程度,方 ...
- [JS设计模式]:工厂模式(3)
简单工厂模式是由一个方法来决定到底要创建哪个类的实例, 而这些实例经常都拥有相同的接口. 这种模式主要用在所实例化的类型在编译期并不能确定, 而是在执行期决定的情况. 说的通俗点,就像公司茶水间的饮料 ...
随机推荐
- cogs 293. [NOI 2000] 单词查找树 Trie树字典树
293. [NOI 2000] 单词查找树 ★★☆ 输入文件:trie.in 输出文件:trie.out 简单对比时间限制:1 s 内存限制:128 MB 在进行文法分析的时候,通常需 ...
- MST + 树形 dp
Genghis Khan(成吉思汗)(1162-1227), also known by his birth name Temujin(铁木真) and temple name Taizu(元太祖), ...
- CheckStyle报错的常见问题及解决方式
CheckStyle报错的常见问题及解决方式 声明: 本文摘自百度文库.希望这篇文章提到的规范能对大家编程起到好的效果,此文不定期更新,将推出更加详尽的编程规范. 1 提示:Type is mis ...
- java反序列化php序列化的对象
最近工作中遇到一个需要解析php序列化后存入DB的array, a:4:{i:0;a:2:{s:11:"province";s:8:"0016";s:7:&qu ...
- AVR单片机教程——小结
本文隶属于AVR单片机教程系列. 第一期挺让我失望的,是我太菜,没有把想讲的都讲出来.经常写了很多,然后一点一点删掉,最后就没多少了. 而且感觉难度不合适,处于很尴尬的位置.讲得简单,难的丢给库, ...
- 【WPF学习】第十章 WPF布局示例
前几章用了相当大的篇幅研究有关WPF布局容器的复杂内容.在掌握了这些基础知识后,就可以研究几个完整的布局示例.通过研究完整的布局示例,可更好的理解各种WPF布局概念在实际窗口中的工作方式. 一.列设置 ...
- 解决luajit ffi cdata引用cdata的问题
使用luajit ffi会遇到cdata引用cdata的情况.官方说明是必须手动保存所有cdata的引用,否则会被gc掉. ffi.cdef[[ struct A { int id; }; struc ...
- 网络io模型总结
操作系统基本概念 首先来来说下操作系统,嗯,操作系统是计算机硬件的管理软件,是对计算机硬件的抽象,操作系统将应用程序分为用户态和内核态,例如驱动程序就位于内核态,而我们写的一般程序都是用户态,包括we ...
- ThreeJS 物理材质shader源码分析(顶点着色器)
再此之前推荐一款GLTF物理材质在线编辑器https://tinygltf.xyz/ ThreeJS 物理材质shader源码分析(顶点着色器) Threejs将shader代码分为ShaderLib ...
- Java 添加、读取、删除Excel形状
本文介绍通过java程序在excel中操作形状(图形)的方法,包括: 1. 添加形状(如设置形状类型/位置/大小.形状颜色填充(单色/渐变色/纹理/图片填充).形状显示或隐藏.形状倾斜角度.添加文本到 ...