1.工厂模式概念
实例化对象,用工厂方法代替new操作(重点)
工厂模式包括工厂方法模式和抽象工厂模式
抽象工厂模式是工厂方法模式的扩展2.什么情况下适合工厂模式
有一组类似的对象需要创建
在编码时不能预见需要创建哪种类的实例
系统需要考虑扩展性,不应依赖于产品类实例如果被创建、组合和表达的细节
根据类图可以知道需要准备的类:
1.人类接口和对应实现
package factory1;
/**
* 定义一个人类接口
* 首先定义什么是人类
*
* @author 1
*/
public interface Human {
/**
* 人是愉快的,会笑的,本来是想用smile表示,想了一下laugh更合适,好长时间没有大笑了;
*/
public void laugh();
/**
* 人类还会哭,代表痛苦
*/
public void cry();
/**
* 人类会说话
*/
public void talk();
}
package factory1;
/**
* 黄种人
*
* @author 1
* @version 1.0
* @description: TODO
* @date 2023/8/15 17:15
*/
public class YellowHuman implements Human {
@Override
public void laugh() {
System.out.println("黄色人类会大笑,幸福呀!");
}
@Override
public void cry() {
System.out.println("黄色人类会哭");
}
@Override
public void talk() {
System.out.println("黄色人类会说话,一般说的都是双字节");
}
}
package factory1;
/**
* 白种人
*
* @author 1
* @version 1.0
* @description: TODO
* @date 2023/8/15 17:16
*/
public class WhiteHuman implements Human {
@Override
public void laugh() {
System.out.println("白色人类会大笑,侵略的笑声");
}
@Override
public void cry() {
System.out.println("白色人类会哭");
}
@Override
public void talk() {
System.out.println("白色人类会说话,一般都是但是单字节!");
}
}
package factory1;
/**
* 黑人
*
* @author 1
* @version 1.0
* @description: TODO
* @date 2023/8/15 17:17
*/
public class BlackHuman implements Human {
@Override
public void laugh() {
System.out.println("黑人会笑");
}
@Override
public void cry() {
System.out.println("黑人会哭");
}
@Override
public void talk() {
System.out.println("黑人可以说话,一般人听不懂");
}
}
2.八卦炉
package factory1;
import java.util.List;
import java.util.Random;
/**
* 造人工厂
* 女娲造人的故事:
*
* @author 1
* @version 1.0
* @description: 很久很久以前,盘古开辟了天地,用身躯造出日月星辰、山川草木,天地一片繁华
* 一天,女娲下界走了一遭,哎!太寂寞,太孤独了,没个会笑的、会哭的、会说话的东东
* 那怎么办呢?不用愁,女娲,神仙呀,造出来呀,然后捏泥巴,放八卦炉(后来这个成了太白金星的宝
* 贝)中烤,于是就有了人:
* 我们把这个生产人的过程用Java程序表现出来:
* @date 2023/8/15 17:19
*/
public class HumanFactory {
/**
* 定义一个八卦炉子
*
* @param c
* @return
*/
public static Human createHuman(Class c) {
//定义一个类型的人类
Human human = null;
try {
//生产一个人类
human = (Human) Class.forName(c.getName()).newInstance();
} catch (InstantiationException e) {
System.out.println("必须指定人类颜色");
} catch (IllegalAccessException e) {
System.out.println("定义的人类错误");
} catch (ClassNotFoundException e) {
System.out.println("定义的人类找不到");
}
return human;
}
/**
* 女娲生气了,把一团泥巴塞到八卦炉,哎产生啥人类就啥人类
*
* @return
*/
public static Human createHuman() {
//定义一个类型的人类
Human human = null;
//首先是获得有多少个实现类,多少个人类
//定义了多少人类
List concreteHumanList =
ClassUtils.getAllClassByInterface(Human.class);
//八卦炉自己开始想烧出什么人就什么人
Random random = new Random();
int rand = random.nextInt(concreteHumanList.size());
human = createHuman(concreteHumanList.get(rand));
return human;
}
}
批量工具类:主要实现根据接口查找实现类(属于扩展知识)
package factory1;
import java.io.File;
import java.io.IOException;
import java.net.URL;
import java.util.ArrayList;
import java.util.Enumeration;
import java.util.List;
/**
* @author 1
* @version 1.0
* @description: TODO
* @date 2023/8/15 17:48
*/
@SuppressWarnings("all")
public class ClassUtils {
//给一个接口,返回这个接口的所有实现类
public static List getAllClassByInterface(Class c){
List returnClassList = new ArrayList(); //返回结果
//如果不是一个接口,则不做处理
if(c.isInterface()){
String packageName = c.getPackage().getName(); //获得当前的包名
try {
//获得当前包下以及子包下的所有类
List allClass = getClasses(packageName);
//判断是否是同一个接口
for(int i=0;i getClasses(String packageName)
throws ClassNotFoundException, IOException {
ClassLoader classLoader = Thread.currentThread()
.getContextClassLoader();
String path = packageName.replace('.', '/');
Enumeration resources = classLoader.getResources(path);
List dirs = new ArrayList();
while (resources.hasMoreElements()) {
URL resource = resources.nextElement();
dirs.add(new File(resource.getFile()));
}
ArrayList classes = new ArrayList();
for (File directory : dirs) {
classes.addAll(findClasses(directory, packageName));
}
return classes;
}
private static List findClasses(File directory, String packageName)
throws ClassNotFoundException {
List classes = new ArrayList();
if (!directory.exists()) {
return classes;
}
File[] files = directory.listFiles();
for (File file : files) {
if (file.isDirectory()) {
assert !file.getName().contains(".");
classes.addAll(findClasses(file, packageName + "." +
file.getName()));
} else if (file.getName().endsWith(".class")) {
classes.add(Class.forName(packageName + '.' +
file.getName().substring(0, file.getName().length() - 6)));
}
}
return classes;
}
}
女娲类
package factory1;
/**
* 女娲
*
* @author 1
* @version 1.0
* @description: TODO
* @date 2023/8/15 17:28
*/
public class NvWa {
public static void main(String[] args) {
System.out.println("(1)女娲第一次造人,火候不足,白人----------------");
Human witheHuman = HumanFactory.createHuman(WhiteHuman.class);
witheHuman.talk();
witheHuman.laugh();
witheHuman.cry();
System.out.println("(2)女娲第二次造人,火候加足点,然后又出了个次品,黑人------------------");
Human blackHuman = HumanFactory.createHuman(BlackHuman.class);
blackHuman.cry();
blackHuman.laugh();
blackHuman.talk();
System.out.println("(3)女娲第三次造人,这次火候掌握的正好,黄色人类-----------------");
Human yellowHuman = HumanFactory.createHuman(YellowHuman.class);
yellowHuman.cry();
yellowHuman.laugh();
yellowHuman.talk();
System.out.println("(4)女娲让八卦炉随机造人------------------");
for (int i = 0; i < 10; i++) {
System.out.println("随机产生人类*******" + i);
Human human = HumanFactory.createHuman();
human.cry();
human.laugh();
human.talk();
}
}
}
运行结果
(1)女娲第一次造人,火候不足,白人----------------
白色人类会说话,一般都是但是单字节!
白色人类会大笑,侵略的笑声
白色人类会哭
(2)女娲第二次造人,火候加足点,然后又出了个次品,黑人------------------
黑人会哭
黑人会笑
黑人可以说话,一般人听不懂
(3)女娲第三次造人,这次火候掌握的正好,黄色人类-----------------
黄色人类会哭
黄色人类会大笑,幸福呀!
黄色人类会说话,一般说的都是双字节
(4)女娲让八卦炉随机造人------------------
随机产生人类*******0
白色人类会哭
白色人类会大笑,侵略的笑声
白色人类会说话,一般都是但是单字节!
随机产生人类*******1
黄色人类会哭
黄色人类会大笑,幸福呀!
黄色人类会说话,一般说的都是双字节
随机产生人类*******2
白色人类会哭
白色人类会大笑,侵略的笑声
白色人类会说话,一般都是但是单字节!
随机产生人类*******3
黑人会哭
黑人会笑
黑人可以说话,一般人听不懂
随机产生人类*******4
黑人会哭
黑人会笑
黑人可以说话,一般人听不懂
随机产生人类*******5
黑人会哭
黑人会笑
黑人可以说话,一般人听不懂
随机产生人类*******6
白色人类会哭
白色人类会大笑,侵略的笑声
白色人类会说话,一般都是但是单字节!
随机产生人类*******7
黑人会哭
黑人会笑
黑人可以说话,一般人听不懂
随机产生人类*******8
黑人会哭
黑人会笑
黑人可以说话,一般人听不懂
随机产生人类*******9
白色人类会哭
白色人类会大笑,侵略的笑声
白色人类会说话,一般都是但是单字节!