设计模式之——工厂模式(A)
本文是在学习中的总结,欢迎转载但请注明出处:http://blog.csdn.net/pistolove/article/details/41085085
昨天看完了工厂模式,觉得在开发的过程中好多地方都能够使用工厂模式,但是很多时候却都没有去使用。我觉得一方面可能是偷懒,懒得去思考,懒得去设计接口;另一方面可能是只为了眼前的需求,觉得开发的产品能使用就行,而不管其后续的维护以及更新。这两个方面长期下去,对自己的提升没有什么好处。其实我自己就是属于第二类,只为了眼前的需求而不去思考后续的维护和更新。可能是刚工作不久,没有那么好的技术,我想我会慢慢转变自己,能够写出漂亮的代码。闲话不多说了,下面分享工厂模式的一些学习总结。
我们一般创建对象都喜欢new一个,因为比较方便;但是我们在使用new的时候,是在实例化一个具体的类,代码绑定了具体的类会导致代码比较脆弱,缺乏弹性;当类发生变化时,涉及到该类的地方都必须修改代码,这样会导致代码一团糟。
假设我们有一家面包店BreadStore,刚开业的时候,只提供一种面包,当需要订购面包的时候,代码可能会这样写:
Bread orderBread(){
Bread bread = new Bread();
bread.prepare();
bread.bake();
bread.cut();
bread.box();
return bread;
}
随着时间的推移,需求变大了,有的人想吃玉米面包,有的人想吃豆沙面包,还有的人想吃巧克力面包......,此时代码可能又变成这样了:
Bread orderBread(String type){
Bread bread;
if("corn".equals(type)){
bread = new CornBread();
} else if("beanpaste".equals(type)){
bread = new BeanpasteBread();
} else if("chocolate".equals(type)){
bread = new ChocolateBread();
}
bread.prepare();
bread.bake();
bread.cut();
bread.box();
return bread;
}
又过了一段时间,附近开了好多家面包店,竞争激烈,你发现竞争者的菜单中多了一些流行风味的面包,比如欧洲面包、荞麦面包,而最近玉米面包又卖的不好,我们决定将其从菜单上撤销,此时的代码又变成这样了:
Bread orderBread(String type){
Bread bread;
if(<span style="color:#cc0000;">"corn".equals(type</span>)){
<span style="color:#ff0000;">bread = new CornBread()</span>;
} else if("beanpaste".equals(type)){
bread = new BeanpasteBread();
} else if("chocolate".equals(type)){
bread = new ChocolateBread();
} <span style="color:#cc33cc;">else if("Europe".equals(type)</span>){
<span style="color:#cc33cc;">bread = new EuropeBread()</span>;
} <span style="color:#cc33cc;">else if("Buckwheat".equals(type)</span>){
<span style="color:#cc33cc;">bread = new BuckwheatBread()</span>;
}
bread.prepare();
bread.bake();
bread.cut();
bread.box();
return bread;
}
我们可以看出,只要随着时间的改变,if...else...里面的需求总是在变化,当我们依赖了具体变化的类时,代码将变得特别难以维护,此时,我们应该考虑到抽象。另外,我们注意到,return上面的四行代码是不想改变的地方,其包含了面包的准备、烘烤、包装,这些流程多年都没有变化,此时可使用封装,将不变的部分封装起来。
首先,我们把创建Bread对象的代码从orderBread()方法中抽离出来,然后把该代码转移到另外一个新对象中,新对象只管如何创建面包。任何时候需要创建面包,就找它。我们也称该对象为“工厂”。这里,我们创建一个简单的面包工厂SimpleBreadFactory,而orderBread()就成为了该工厂的客户。当需要面包的时候,就叫面包工厂做一个出来。现在,orderBread()方法只关心从工厂得到一个面包,而此时的面包实现了抽象的Bread接口,而Bread接口提供了prepare()、bake()、cut()、box()方法来进行相应操作。抽象Bread和创建的简单工厂的代码如下:
public abstract class Bread {
void prepare(){
System.out.println("prepare for...");
}
void bake() {
System.out.println("Bake for 25 minutes at 350");
}
void cut() {
System.out.println("Cutting the pizza into diagonal slices");
}
void box() {
System.out.println("Place pizza in official PizzaStore box");
}
}
public class SimpleBreadFactory(){
public Bread() creatBread(String type){
Bread bread = null;
if("corn".equals(type)){
bread = new CornBread();
} else if("beanpaste".equals(type)){
bread = new BeanpasteBread();
} else if("chocolate".equals(type)){
bread = new ChocolateBread();
}
return bread;
}
}
现在,我们可以重新构造BreadStore了,我们要做的就是依靠工厂来为我们创建面包,代码如下:
public class BreadStore{
SimpleBreadFactory factory;
public BreadStore(SimpleBreadFactory factory){
this.factory = factory;
}
public Bread orderBread(String type){
Bread bread; /**未抽象之前的Bread**/
bread = factory.creatBread(type);
bread.prepare();
bread.bake();
bread.cut();
bread.box();
return bread;
}
}
在上面的代码中,我们为BreadStore加上了对简单面包工厂的引用;在BreadStore是构造器中需要传入一个工厂作为参数;在orderBread()方法中通过简单传入订单类型来使用工厂创建面包,在这里面我们把new操作符替换为工厂对象创建方法,不再使用具体实例化。
若干年后,面包店经营的不错,很多地方的消费者都希望面包店能够在当地有加盟店,此时,简单面包工厂已经不能满足需求了,加盟店都希望有适合当地需求的面包工厂,这样就产生了类似BJBreadFactory、TJBreadFactory、SHBreadFactory的面包工厂。
BJBreadFactory bjFactory = new BJBreadFactory();
BreadStore bjStore = new BreadStore(bjFactory);
bjStore.orderBread("bjtaste");
TJBreadFactory tjFactory = new TJBreadFactory();
BreadStore tjStore = new BreadStore(tjFactory);
tjStore.orderBread("bjtaste");
SHBreadFactory shFactory = new SHBreadFactory();
BreadStore shStore = new BreadStore(shFactory);
shStore.orderBread("bjtaste");
考虑到让面包制作局限于BreadStore,同时又能够让加盟店仍然可以自由制作该区的面包,这样就允许子类做决定了。我们将BreadStore抽象出来。
public abstract class BreadStore{
abstract Bread creatBread(String type);
public Bread orderBread(String type){
Bread bread; /**未抽象之前的Bread**/
bread = factory.creatBread(type);
bread.prepare();
bread.bake();
bread.cut();
bread.box();
return bread;
}
}
对于BreadStore的子类,每一个都会覆盖createBread()方法,同时又使用BreadStore中的orderBread()方法,其实也可将orderBread()声明为final,以防止被子类覆盖。这样,不同地区的加盟店都可以创建需要的面包了。
我们在武汉开一家新的面包店:
public class WHBreadStore extends BreadStore{
Bread creatBread(String item){
if("corn".equals(item)){
bread = new WHCornBread();
} else if("beanpaste".equals(item)){
bread = new WHBeanpasteBread();
} else if("chocolate".equals(item)){
bread = new WHChocolateBread();
} else return null;
}
}
通过声明工厂方法 ,我们可以看出:原来是由一个对象负责所有具体类的实例化,现在我们对BreadStore做了一些调整,变成了由一群子类来负责实例化。
对于 abstract Product factoryMethod(String type)工厂方法:
(1)工厂方法是抽象的,所以需要依赖子类来处理对象的创建。
(2)工厂方法必须返回一个产品。在超类中定义的方法,通常会用到工厂方法的返回值。
(3)工厂方法将客户(超类中的代码,如orderBread()和实际创建具体产品的代码分离开来)。
最后,我们再来认识一下工厂方法模式:所有的工厂模式都是用来封装对象的创建,这样就减小了耦合。它是通过让子类来决定该创建的对象是什么,从而来达到将对象创建的过程封装的目的。下面两张图反映了创建者类和产品类之间的关系:


设计模式之——工厂模式(A)的更多相关文章
- 设计模式——抽象工厂模式及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)
简单工厂模式是由一个方法来决定到底要创建哪个类的实例, 而这些实例经常都拥有相同的接口. 这种模式主要用在所实例化的类型在编译期并不能确定, 而是在执行期决定的情况. 说的通俗点,就像公司茶水间的饮料 ...
随机推荐
- java求素数
按定义 即除了1和它本身以外不再被其他的除数整数 public static void main(String[] args) { for (int i = 2; i < 100; i++) { ...
- Python中的转义
在Python交互式解释器中,输出的字符串会用引号引起来,特殊字符会用反斜杠(\)转义.如果遇到带有\的字符被当作特殊字符时,有以下两种处理方法:1.使用双反斜杠(\\)来转义2.使用原始字符串,方法 ...
- opencv视屏流嵌入wxpython框架
前几篇博客分享搭建人脸识别与情绪判断的环境和源码,但是没有UI,界面很难看,一打开就是opencv弹出的一个视屏框.处女座的我看着非常难受,于是决定做一个UI,稍微规矩好看一点,再怎么说,这样的话也算 ...
- beta 分布的详细介绍(转载)
目前看到的关于beta分布最好的一个解释,由于贴过来格式不好看,所以附上链接: http://www.datalearner.com/blog/1051505532393058
- 验证码识别之w3cschool字符图片验证码(easy级别)
起因: 最近在练习解析验证码,看到了这个网站的验证码比较简单,于是就拿来解析一下攒攒经验值,并无任何冒犯之意... 验证码所在网页: https://www.w3cschool.cn/checkmph ...
- Python 元组内置函数
Python元组包含了以下内置函数 序号 方法及描述 1 cmp(tuple1, tuple2)比较两个元组元素. 2 len(tuple)计算元组元素个数. 3 max(tuple)返回元组中元素最 ...
- ROS新功能包PlotJuggler绘图
http://www.ros.org/news/2017/01/new-package-plotjuggler.html PlotJuggler,一个基于Qt的应用程序,允许用户加载,搜索和绘图数据. ...
- Lua语言模型 与 Redis应用
Lua语言模型 与 Redis应用 标签: Java与NoSQL 从 2.6版本 起, Redis 开始支持 Lua 脚本 让开发者自己扩展 Redis. 本篇博客主要介绍了 Lua 语言不一样的设计 ...
- 我的第一个RootKit,支持XP、Vista、Win7、Win8 RTM 32位
只有写过一个BootKit,才能比较深刻的理解其整个过程与机制,也能加深对Windows系统引导各个过程的熟悉和理解. 我写的这个bootkit,暂时还没想到一个比较好的名字,它 1. 支持xp到w ...
- Android开发学习之路--图表实现(achartengine/MPAndroidChart)之初体验
已经有一段时间没有更新博客了,在上周离开工作了4年的公司,从此不再安安稳稳地工作了,更多的是接受挑战和实现自身价值的提高.离开了嵌入式linux,从此拥抱移动互联网,也许有点为时已晚,但是相信通过 ...