工厂方法模式(java 设计模式)
1.工厂方法模式的定义
工厂方法模式使用的频率非常高, 在我们日常的开发中总能见到它的身影。 其定义为:
Define an interface for creating an object,but let subclasses decide which class to
instantiate.Factory Method lets a class defer instantiation to subclasses.( 定义一个用于创建对象的
接口, 让子类决定实例化哪一个类。 工厂方法使一个类的实例化延迟到其子类。 )
2.工厂方法模式的通用代码
抽象产品类
public abstract class Product {
//产品类的公共方法
public void method1(){
//业务逻辑处理
}
//抽象方法
public abstract void method2();
}
具体的产品类可以有多个, 都继承于抽象产品类, 其源代码如代码清单8-9所示。
具体产品类
public class ConcreteProduct1 extends Product {
public void method2() {
//业务逻辑处理
}
}
public class ConcreteProduct2 extends Product {
public void method2() {
//业务逻辑处理
}
}
抽象工厂类负责定义产品对象的产生, 源代码如代码清单8-10所示。
抽象工厂类
public abstract class Creator {
/*
* 创建一个产品对象, 其输入参数类型可以自行设置
* 通常为String、 Enum、 Class等, 当然也可以为空
*/
public abstract <T extends Product> T createProduct(Class<T> c);
}
具体如何产生一个产品的对象, 是由具体的工厂类实现的,
具体工厂类
public class ConcreteCreator extends Creator {
public <T extends Product> T createProduct(Class<T> c){
Product product=null;
try {
product = (Product)Class.forName(c.getName()).newInstance();
} catch (Exception e) {
//异常处理
}
return (T)product;
}
}
场景类
public class Client {
public static void main(String[] args) {
Creator creator = new ConcreteCreator();
Product product = creator.createProduct(ConcreteProduct1.class);
/*
* 继续业务处理
*/
}
}
3.工厂方法模式实例
package ch08;
public interface Human {
// 人都有肤色
public void getColor();
// 人都分语言
public void getTalk();
}
package ch08;
public class BlackHuman implements Human {
public void getColor() {
// TODO Auto-generated method stub
System.out.println("黑人肤色为黑色");
}
public void getTalk() {
// TODO Auto-generated method stub
System.out.println("黑人说话一般听不懂");
}
}
package ch08;
public class WhiteHuman implements Human{
public void getColor() {
// TODO Auto-generated method stub
System.out.println("白人肤色为白");
}
public void getTalk() {
// TODO Auto-generated method stub
System.out.println("白人一般说英语");
}
}
package ch08;
public class YellowHuman implements Human{
public void getColor() {
// TODO Auto-generated method stub
System.out.println("黄色人种肤色为黄色");
}
public void getTalk() {
// TODO Auto-generated method stub
System.out.println("黄色人种一般说汉语");
}
}
package ch08;
public abstract class AbstractHumanFactory {
//泛型 通过定义泛型对creatHuman的输入参数产生两层限制: 必须是Class类型;必须是Human的实现类。
public abstract <T extends Human> T createHuman(Class<T> c);
// 泛型类可看作普通类的工厂
/**
*
* @author
* 泛型方法,定义一个带有类型参数的简单方法
* 这个方法是在普通类中定义的,而不是在泛型类中定义的。然而,这是一个泛型方法,可
以从尖括号和类型变量看出这一点。注意,类型变量放在修饰符(这里是public static)的后面,
返回类型的前面。
*/
/*class ArrayAlg
{
public static <T> T getMiddle(T... a)
{
return a[a.length / 2];
}
}*/
}
package ch08;
public class HumanFactory extends AbstractHumanFactory{
public <T extends Human> T createHuman(Class<T> c) {
// TODO Auto-generated method stub
//定义一个生产的人种;
Human human = null;
try {
/* 产生一个人种, 通过Class.forName(c.getName()).newInstance()实例化一个
c对象然后再把c对象强制转化为Human或T类型的对象*/
/*Class.forName(xxx.xx.xx) 返回的是一个类, .newInstance() 后才创建一个对象*/
// human = (T) Class.forName(c.getName()).newInstance();//也对
human = (Human) Class.forName(c.getName()).newInstance();
} catch (InstantiationException e) {
// TODO Auto-generated catch block
e.printStackTrace();
} catch (IllegalAccessException e) {
// TODO Auto-generated catch block
e.printStackTrace();
} catch (ClassNotFoundException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
return (T)human;
}
}
package ch08;
//女娲类
public class NvWa {
public static void main(String[] args) {
//声明一个造人的阴阳八卦炉
AbstractHumanFactory YingYangLu = new HumanFactory();
//女娲第一次造人,火候不足,于是白人产生了
System.out.println("--造出的第一批人是白色人种--");
// 现在, Class类是泛型的。 WhiteHuman.class实际上是一个Class<WhiteHuman>(应该是Class<Human>类的对象
// (事实上,是惟一的对象)。
Human whiteHuman = YingYangLu.createHuman(WhiteHuman.class); whiteHuman.getColor();
whiteHuman.getTalk();
System.out.println("--造出的第二批人是黑色人种--");
Human blackHuman = YingYangLu.createHuman(BlackHuman.class);
blackHuman.getColor();
blackHuman.getTalk();
//第三次造人,火候刚刚好,于是黄色人种产生了
System.out.println("--造出的第三批人是黄色人种--");
Human yellowHuman = YingYangLu.createHuman(YellowHuman.class);
yellowHuman.getColor();
yellowHuman.getTalk(); }
}
4.工厂方法模式的优点
首先, 良好的封装性, 代码结构清晰。 一个对象创建是有条件约束的, 如一个调用者需
要一个具体的产品对象, 只要知道这个产品的类名( 或约束字符串) 就可以了, 不用知道创
建对象的艰辛过程, 降低模块间的耦合。
其次, 工厂方法模式的扩展性非常优秀。 在增加产品类的情况下, 只要适当地修改具体
的工厂类或扩展一个工厂类, 就可以完成“拥抱变化”。 例如在我们的例子中, 需要增加一个
棕色人种, 则只需要增加一个BrownHuman类, 工厂类不用任何修改就可完成系统扩展。
再次, 屏蔽产品类。 这一特点非常重要, 产品类的实现如何变化, 调用者都不需要关
心, 它只需要关心产品的接口, 只要接口保持不变, 系统中的上层模块就不要发生变化。 因
为产品类的实例化工作是由工厂类负责的, 一个产品对象具体由哪一个产品生成是由工厂类
决定的。 在数据库开发中, 大家应该能够深刻体会到工厂方法模式的好处: 如果使用JDBC
连接数据库, 数据库从MySQL切换到Oracle, 需要改动的地方就是切换一下驱动名称( 前提
条件是SQL语句是标准语句) , 其他的都不需要修改, 这是工厂方法模式灵活性的一个直接
案例。
工厂方法模式(java 设计模式)的更多相关文章
- 设计模式之第1章-工厂方法模式(Java实现)
设计模式之第1章-工厂方法模式(Java实现) “我先来”,“不,老公,我先!”.远远的就听到几个人,哦不,是工厂方法模式和抽象工厂模式俩小夫妻在争吵,尼妹,又不是吃东西,谁先来不都一样(吃货的世界~ ...
- 设计模式之工厂方法模式(Java实现)
“我先来”,“不,老公,我先!”.远远的就听到几个人,哦不,是工厂方法模式和抽象工厂模式俩小夫妻在争吵,尼妹,又不是吃东西,谁先来不都一样(吃货的世界~).“抽象工厂模式,赶紧的自我介绍,工厂方法模式 ...
- 工厂方法模式--java代码实现
工厂方法模式 工厂方法模式,对简单工厂模式进行了升级.我们将水果园比作一个工厂,在简单工厂模式下,水果园是一个具体的工厂,直接用来生产各种各样的水果.那么在工厂方法模式下,水果园是一个抽象工厂,那么苹 ...
- FactoryMethodPattern(工厂方法模式)-----Java/.Net
也就是工厂方法(FactoryMethod)模式允许将产品类的实例化推迟到具体的创建者子类,由创建者子类决定实例化哪一个产品类.我们同样以汽车的生产作为讲解该模式的例子,因为汽车生产从宏观上来说也是特 ...
- 工厂方法模式(JAVA反射)
简单工厂模式的不足 在简单工厂模式中,只提供了一个工厂类,该工厂类处于对产品类进行实例化的中心位置,它知道每一个产品对象的创建细节,并决定何时实例化哪一个产品类.简单工厂模式最大的缺点是当有新 ...
- Java设计模式(二) 工厂方法模式
本文介绍了工厂方法模式的概念,优缺点,实现方式,UML类图,并介绍了工厂方法(未)遵循的OOP原则 原创文章.同步自作者个人博客 http://www.jasongj.com/design_patte ...
- JAVA设计模式--工厂方法模式
工厂方法设计模式 抽象工厂角色: 这是工厂方法模式的核心,它与应用程序无关.是具体工厂角色必须实现的接口或者必须继承的父类.在java中它由抽象类或者接口来实现.具体工厂角色:它含有和具体业务逻辑有关 ...
- java 23 - 1 设计模式之工厂方法模式
转载: JAVA设计模式之工厂模式(简单工厂模式+工厂方法模式)
- Java设计模式之工厂模式(简单工厂模式+工厂方法模式)
摘自http://blog.csdn.net/jason0539/article/details/23020989 在面向对象编程中, 最通常的方法是一个new操作符产生一个对象实例,new操作符就是 ...
- 我的Java设计模式-工厂方法模式
女朋友dodo闹脾气,气势汹汹的说"我要吃雪糕".笔者心里啊乐滋滋的,一支雪糕就能哄回来,不亦乐乎?! 但是,雪糕买回来了,她竟然说"不想吃雪糕了,突然想吃披萨" ...
随机推荐
- HDU 3721 Building Roads (2010 Asia Tianjin Regional Contest) - from lanshui_Yang
感慨一下,区域赛的题目果然很费脑啊!!不过确实是一道不可多得的好题目!! 题目大意:给你一棵有n个节点的树,让你移动树中一条边的位置,即将这条边连接到任意两个顶点(边的大小不变),要求使得到的新树的直 ...
- C#中的枚举器(转)
术语表 Iterator:枚举器(迭代器) 如果你正在创建一个表现和行为都类似于集合的类,允许类的用户使用foreach语句对集合中的成员进行枚举将会是很方便的.这在C# 2.0中比 C# 1.1更容 ...
- 一次不是事故的SSH闪断问题
从前一天下午的一个瞬间,公司内所有的ssh 连接在没有任何征兆的情况下,全部开始闪断. 折腾了一天,关闭过SELinux, 清空过Iptables,软硬重启过服务器,交换机,路由,重新配置过sshd文 ...
- Intent携带额外的数据的方法
1.putExtras(Bundle data):向Intent中放入需要“携带”的数据.2.putXxx(String key,Xxx data):向Bundle放入Int.Long等各种类型的数据 ...
- 51单片机C语言学习笔记4:keil C51绝对地址访问
在利用keil进行8051单片机编程的时,常常需要进行绝对地址进行访问.特别是对硬件操作,如DA AD 采样 ,LCD 液晶操作,打印操作.等等.C51提供了三种访问绝对地址的方法: 1. 绝对宏: ...
- 【笔试&面试】C#中的程序集
1. C#中的程序集(Assembly) 答:程序集是包含一个或多个类型定义文件和资源文件的集合.它允许我们分离可重用类型的逻辑表示和物理表示. 程序集是一个可重用.可实施版本策略和安全策略 ...
- 注意:rsyslog 源码安装 会出现日志重复发的情况,需要rpm包安装
cd /etc/yum.repos.d;wget http://rpms.adiscon.com/v8-stable/rsyslog.repo uat-web02:/etc/yum.repos.d# ...
- BZOJ3391: [Usaco2004 Dec]Tree Cutting网络破坏
3391: [Usaco2004 Dec]Tree Cutting网络破坏 Time Limit: 1 Sec Memory Limit: 128 MBSubmit: 47 Solved: 37[ ...
- 【转】Beagleboard:BeagleBoneBlack
原文网址:http://elinux.org/Beagleboard:BeagleBoneBlack Did you know that elinux.org has Mailing Lists? P ...
- SDL2.0教程翻译·目录
原文地址:SDL 2.0 Tutorial Index Welcome! 下面的教程旨在为你提供一个SDL2.0以及c++中游戏设计和相关概念的介绍.在本教程中,我们假定你对C++有一定程度上的知识, ...