GoF23:工厂模式(Factory)

核心本质

  1. 实例化对象不使用new,用工厂方法代替
  2. 将选择实现类,创建对象统一管理和控制。从而将调用者跟我们的实现类解耦

工厂模式三种模式

  1. 简单工厂模式(静态工厂模式)
  2. 工厂方法模式
  3. 抽象工厂模式

简单工厂模式(静态工厂模式)

用来生产同一等级结构中的任意产品(对于增加新的产品,需要扩展已有的代码,不符合开闭原则)

​ 现在定义一种情景,就是消费者去买车,他可以自己组成车,也可以找一个工厂来生产几种不同类型的车,也可以到专门生产该车型的工厂买。

  • 定义一个接口
  1. public interface Car {
  2. void getName();
  3. }
  • 让两种不同的汽车类都去实现Car接口
  1. public class Tesla implements Car{
  2. @Override
  3. public void getName() {
  4. System.out.println("特斯拉汽车");
  5. }
  6. }
  1. public class WuLing implements Car{
  2. @Override
  3. public void getName() {
  4. System.out.println("五菱汽车");
  5. }
  6. }

在Consumer的主方法中的操作相当于是我们自己组成的对象(自己组车):

  1. public class Consumer {
  2. public static void main(String[] args) {
  3. // 原先我们要使用哪个对象,都是我们自己new出来的,相当于是自己创建的
  4. Car car1 = new WuLing();
  5. Car car2 = new Tesla();
  6. car1.getName();
  7. car2.getName();
  8. }
  9. }

现在我们让一个工厂来生产我们所需要的对象,不需要我们自己去new了:

  1. /**
  2. * 简单工厂模式,并不符合开闭原则,如果新增产品,就必须对原先的代码进行修改
  3. */
  4. public class CarFactory {
  5. // 方式一:
  6. public static Car getCar(String name) {
  7. if ("五菱".equals(name)) {
  8. return new WuLing();
  9. } else if ("特斯拉".equals(name)) {
  10. return new Tesla();
  11. } else {
  12. return null;
  13. }
  14. }
  15. // 方式二:
  16. public static Car getWuLing() {
  17. return new WuLing();
  18. }
  19. public static Car getTesla() {
  20. return new Tesla();
  21. }
  22. }

在Consumer的主方法中的操作相当于从工厂中获取我们需要的对象:

  1. public class Consumer {
  2. public static void main(String[] args) {
  3. // 现在使用简单工厂模式,来获取我们需要的对象
  4. Car car1 = CarFactory.getCar("五菱");
  5. Car car2 = CarFactory.getCar("特斯拉");
  6. car1.getName();
  7. car2.getName();
  8. }
  9. }

但是上面的简单工厂有一些弊端,如果我们需要新增一个类,就需要对工厂中的代码进行修改,这不符合开闭原则(对扩展是开放的,对修改是关闭的)

工厂方法模式

用来生产同一等级结构中的固定产品(支持增加任意)

工厂方法模式还是承接上面的情景,只不过每一个对象,即车,都有一个专门工厂去生产该对象。

  1. public interface Car {
  2. void getName();
  3. }
  • 对原来的CarFactory进行提取,相当于是加上一层
  1. public interface CarFactory {
  2. Car getCar();
  3. }
  1. public class WuLing implements Car {
  2. @Override
  3. public void getName() {
  4. System.out.println("五菱汽车");
  5. }
  6. }
  • 专门生产五菱汽车的工厂
  1. public class WuLingFactory implements CarFactory{
  2. @Override
  3. public Car getCar() {
  4. return new WuLing();
  5. }
  6. }
  1. public class Tesla implements Car {
  2. @Override
  3. public void getName() {
  4. System.out.println("特斯拉汽车");
  5. }
  6. }
  • 专门生产特斯拉汽车的工厂
  1. public class TeslaFactory implements CarFactory{
  2. @Override
  3. public Car getCar() {
  4. return new Tesla();
  5. }
  6. }
  • 不去new我们需要的对象,而是new一个工厂对象去生产我们需要的对象
  1. public class Consumer {
  2. public static void main(String[] args) {
  3. Car car1 = new WuLingFactory().getCar();
  4. Car car2 = new TeslaFactory().getCar();
  5. car1.getName();
  6. car2.getName();
  7. }
  8. }

工厂方法模式就可以避免修改CarFactory中的代码,当我们需要新增一个对象时,只需要创建生产对象的工厂即可。符合开闭原则,但是无论在结构复杂度、管理复杂度……都是simple简单工厂占优势,因此在设计原则上:工厂方法模式,在实际业务中:简单工厂模式。

抽象工厂模式

围绕一个超级工厂创建其他工厂。该超级工厂又称为其他工厂的工厂。

举例说明

  • UML类图:

  • Creator:产品族工厂,定义了一系列的产品生产行为
  • ConcreteCreator:具体的产品族工厂
  • Product:抽象产品接口
  • ProductA1/A2/B1/B2:都是具体的产品,实现了相应的产品接口

举个栗子:

手机是一个抽象的产品,手机有小米手机,华为手机,路由器也是一个抽象的产品,路由器有小米路由器,华为路由器。

具体的手机和抽象手机就构成了一个产品等级结构

具体的路由器和抽象路由器就构成了另外一个产品等级结构

实质上产品等级结构即产品的继承结构,即它们都实现了相同的接口。

划重点就是产品族中的产品都是由同一个工厂生产的,位于不同的产品等级结构。因为同一产品族中的各个产品,实现的不是相同的接口,所以不位于同一产品等级结构。

它们之间的关系如下图所示:

对上图的说明:

  • 图中横坐标代表产品等级,新增产品等级为横坐标的增长,方向向右,具体表现为新增一列;

  • 图中纵坐标为产品族等级,新增产品族为纵坐标的增长,方向向上,具体表现为新增一排;

对比一下工厂方法和抽象工厂方法:

工厂模式针对的是产品等级结构,定义一种生产该产品工厂的接口,让不同类型的厂商去分别实现它,每一个实现了接口的工厂专门用来生产这一种产品。

抽象工厂模式针对的是产品族等级结构,这个抽象工厂是用来生产工厂的,被生产的工厂再去生产它们各自产品族中的产品,这样,每一个实现类工厂都可以生产本产品族的不同产品。

如何选择:

​ 在两者的使用选择,要结合实际业务,对于产品等级数量相对固定的产品族,可以优先考虑抽象工厂模式,但是如果频繁变动,则不大适用,因为在现有的产品族中新增产品等级时,就需要修改产品族工厂,也就违背了开闭原则而对于新增产品族,只需实现相应的接口即可,不需要修改产品族工厂中的代码。

代码实现

这里以上面的手机和路由器为栗:

简要的实现说明:

  • 定义抽象产品接口
  1. //手机产品接口
  2. public interface IPhoneProduct {
  3. //开机
  4. void start();
  5. //关机
  6. void shutdown();
  7. //拨打电话
  8. void callUp();
  9. //发送短信
  10. void sendSMS();
  11. }

  1. //路由器产品接口
  2. public interface IRouterProduct {
  3. //开机
  4. void start();
  5. //关机
  6. void shutdown();
  7. //开启wifi
  8. void openWifi();
  9. //设置参数
  10. void setting();
  11. }
  • 定义小米品牌手机和路由器产品实现类、华为品牌手机和路由器产品实现类
  1. //小米手机产品
  2. public class XiaomiPhone implements IPhoneProduct {
  3. @Override
  4. public void start() {
  5. System.out.println("开启小米手机");
  6. }
  7. @Override
  8. public void shutdown() {
  9. System.out.println("关闭小米手机");
  10. }
  11. @Override
  12. public void callUp() {
  13. System.out.println("用小米手机打电话");
  14. }
  15. @Override
  16. public void sendSMS() {
  17. System.out.println("用小米手机发短信");
  18. }
  19. }
  1. //小米路由器产品
  2. public class XiaomiRouter implements IRouterProduct {
  3. @Override
  4. public void start() {
  5. System.out.println("启动小米路由器");
  6. }
  7. @Override
  8. public void shutdown() {
  9. System.out.println("关闭小米路由器");
  10. }
  11. @Override
  12. public void openWifi() {
  13. System.out.println("打开小米路由器的wifi功能");
  14. }
  15. @Override
  16. public void setting() {
  17. System.out.println("设置小米路由器参数");
  18. }
  19. }
  1. //华为手机产品
  2. public class HuaweiPhone implements IPhoneProduct {
  3. @Override
  4. public void start() {
  5. System.out.println("开启华为手机");
  6. }
  7. @Override
  8. public void shutdown() {
  9. System.out.println("关闭华为手机");
  10. }
  11. @Override
  12. public void callUp() {
  13. System.out.println("用华为手机打电话");
  14. }
  15. @Override
  16. public void sendSMS() {
  17. System.out.println("用华为手机发短信");
  18. }
  19. }
  1. //华为路由器产品
  2. public class HuaweiRouter implements IRouterProduct {
  3. @Override
  4. public void start() {
  5. System.out.println("启动华为路由器");
  6. }
  7. @Override
  8. public void shutdown() {
  9. System.out.println("关闭华为路由器");
  10. }
  11. @Override
  12. public void openWifi() {
  13. System.out.println("打开华为路由器的wifi功能");
  14. }
  15. @Override
  16. public void setting() {
  17. System.out.println("设置华为路由器参数");
  18. }
  19. }
  • 定义一个抽象工厂接口,或者称之为抽象产品族工厂
  1. //抽象产品工厂(定义了同一个产品族的产品生产行为)
  2. public interface IProductFactory {
  3. //生产手机
  4. IPhoneProduct produceTelPhone();
  5. //生产路由器
  6. IRouterProduct produceRouter();
  7. }
  • 小米工厂和华为工厂的实现了
  1. //小米产品工厂
  2. public class XiaomiProductFactory implements IProductFactory {
  3. @Override
  4. public IPhoneProduct produceTelPhone() {
  5. System.out.println(">>>>>>生产小米手机");
  6. return new XiaomiPhone();
  7. }
  8. @Override
  9. public IRouterProduct produceRouter() {
  10. System.out.println(">>>>>>生产小米路由器");
  11. return new XiaomiRouter();
  12. }
  13. }
  1. //华为产品工厂
  2. public class HuaweiProductFactory implements IProductFactory{
  3. @Override
  4. public IPhoneProduct produceTelPhone() {
  5. System.out.println(">>>>>>生产华为手机");
  6. return new HuaweiPhone();
  7. }
  8. @Override
  9. public IRouterProduct produceRouter() {
  10. System.out.println(">>>>>>生产华为路由器");
  11. return new HuaweiRouter();
  12. }
  13. }
  • 客户端
  1. //客户端
  2. public class Client {
  3. public static void main(String[] args) {
  4. System.out.println("===================小米系列产品=================");
  5. //小米产品工厂实例
  6. IProductFactory xiaomiProductFactory = new XiaomiProductFactory();
  7. //生产小米路由器
  8. IRouterProduct xiaomiRouter = xiaomiProductFactory.produceRouter();
  9. xiaomiRouter.start();
  10. xiaomiRouter.setting();
  11. xiaomiRouter.openWifi();
  12. xiaomiRouter.shutdown();
  13. //生产小米手机
  14. IPhoneProduct xiaomiPhone = xiaomiProductFactory.produceTelPhone();
  15. xiaomiPhone.start();
  16. xiaomiPhone.callUp();
  17. xiaomiPhone.sendSMS();
  18. xiaomiPhone.shutdown();
  19. System.out.println("===================华为系列产品=================");
  20. //华为产品工厂实例
  21. IProductFactory huaweiProductFactory = new HuaweiProductFactory();
  22. //生产华为路由器
  23. IRouterProduct huaweiRouter = huaweiProductFactory.produceRouter();
  24. huaweiRouter.start();
  25. huaweiRouter.setting();
  26. huaweiRouter.openWifi();
  27. huaweiRouter.shutdown();
  28. //生产华为手机
  29. IPhoneProduct huaweiPhone = huaweiProductFactory.produceTelPhone();
  30. huaweiPhone.start();
  31. huaweiPhone.callUp();
  32. huaweiPhone.sendSMS();
  33. huaweiPhone.shutdown();
  34. }
  35. }
  • 运行结果:
  1. ===小米系列产品=
  2. >>>>>>生产小米路由器
  3. 启动小米路由器
  4. 设置小米路由器参数
  5. 打开小米路由器的wifi功能
  6. 关闭小米路由器
  7. >>>>>>生产小米手机
  8. 开启小米手机
  9. 用小米手机打电话
  10. 用小米手机发短信
  11. 关闭小米手机
  12. ===华为系列产品=
  13. >>>>>>生产华为路由器
  14. 启动华为路由器
  15. 设置华为路由器参数
  16. 打开华为路由器的wifi功能
  17. 关闭华为路由器
  18. >>>>>>生产华为手机
  19. 开启华为手机
  20. 用华为手机打电话
  21. 用华为手机发短信
  22. 关闭华为手机
  • 关系图:

GoF23:工厂模式(Factory)的更多相关文章

  1. 设计模式(一)工厂模式Factory(创建型)

    设计模式一 工厂模式Factory 在面向对象编程中, 最通常的方法是一个new操作符产生一个对象实例,new操作符就是用来构造对象实例的.可是在一些情况下, new操作符直接生成对象会带来一些问题. ...

  2. 设计模式(一)工厂模式Factory(创建类型)

    设计模式一 工厂模式Factory 在面向对象编程中, 最通常的方法是一个new操作符产生一个对象实例,new操作符就是用来构造对象实例的.可是在一些情况下, new操作符直接生成对象会带来一些问题. ...

  3. 工厂模式(Factory)和抽象工厂模式(Abstract Factory)

    一.工厂模式(Factory):通过让子类决定该创建的对象是什么,来达到将对象创建的过程封装的目的,工厂方法让类的实例化推迟到子类 (1)涉及角色:抽象产品,具体产品,抽象创建者,具体创建者.     ...

  4. 设计模式(一)工厂模式Factory(创建型)(转)

    原文链接:http://blog.csdn.net/hguisu/article/details/7505909 设计模式一 工厂模式Factory 在面向对象编程中, 最通常的方法是一个new操作符 ...

  5. JAVA:将反射技术应用于工厂模式(Factory)和单例模式(Singleton)的简单代码

    反射技术大量用于Java设计模式和框架技术,最常见的设计模式就是工厂模式(Factory)和单例模式(Singleton). 参考URL: http://blog.csdn.net/xiaohai79 ...

  6. 设计模式 - 工厂模式(factory pattern) 具体解释

    版权声明:本文为博主原创文章,未经博主同意不得转载. https://blog.csdn.net/u012515223/article/details/27081511 工厂模式(factory pa ...

  7. 工厂模式 - Factory

    简单工厂模式 SimpleFactory Pattern,将一个具体类的实例化交给一个静态工厂方法来执行. 特点: 增加功能需要修改工厂类,扩展性较差: 参考: 设计模式学习笔记 - 简单工厂模式: ...

  8. 设计模式--工厂模式Factory(创建型)

    工厂模式属于创建型模式,分为三类,简单工厂模式.工厂方法模式.抽象工厂模式. 一.简单工厂模式 在工厂中做判断,根据产品类型从而创造相应的产品,当增加新产品时需要修改工厂类. 例如: enum CTY ...

  9. 工厂模式/factory模式/创建型模式

    工厂模式 普通工厂模式 原本需要new出来的对象,通过一个类的方法去搞定,Factory.build(parameter),类似这种. public interface Sender { public ...

随机推荐

  1. Tomcat5启动流程与配置详解

    标签:配置 tomcat 休闲 职场 原创作品,允许转载,转载时请务必以超链接形式标明文章 原始出处 .作者信息和本声明.否则将追究法律责任.http://zhangjunhd.blog.51cto. ...

  2. jmeter 信息头Bearer

    1.数据规则 2.登录时获取token信息 3.正则表达式获取token值 说明: (1)引用名称:下一个请求要引用的参数名称,如填写title,则可用${title}引用它. (2)正则表达式: ( ...

  3. shell 脚本常用调试方法

    曾经我刚开始学习 shell 脚本时,除了知道用 echo 输出一些信息外,并不知道其他方法,仅仅依赖 echo 来查找错误,比较难调试且过程繁琐.效率低下.本文介绍下我常用的一些 shell 脚本调 ...

  4. mvc传递json数据到view简单实例

    基于extjs4.2 controller //存储数据的类 public class DataLink { public string Name { get; set; } public strin ...

  5. 在众多小说中,Python告诉你哪本小说好看

    前言 本文的文字及图片来源于网络,仅供学习.交流使用,不具有任何商业用途,版权归原作者所有,如有问题请及时联系我们以作处理. 作者: 有趣的Python PS:如有需要Python学习资料的小伙伴可以 ...

  6. L16 LeNet

    **本小节用到的数据下载 1.涉及语句 import d2lzh1981 as d2l 数据1 : d2lzh1981 链接:https://pan.baidu.com/s/1LyaZ84Q4M75G ...

  7. Android | 教你如何在安卓上实现通用卡证识别,一键各种卡绑定

    目录 前言 通用卡证识别的应用场景 如何使用通用卡证识别服务 集成通用卡证识别服务的关键流程 开发实战 1 开发准备 1.1 在项目级gradle里添加华为maven仓 1.2 在应用级的build. ...

  8. 使用 selenium 实现谷歌以图搜图爬虫

    使用selenium实现谷歌以图搜图 实现思路 原理非常简单,就是利用selenium去操作浏览器,获取到想要的链接,然后进行图片的下载,和一般的爬虫无异. 用到的技术:multiprocessing ...

  9. bypass安全狗测试学习

    搭建简单的sql注入环境 在test数据库中创建sqltest表,插入字段数据 编写存在注入的php文件 <?php $id = $_REQUEST['uid']; echo "您当前 ...

  10. Spring5:IOC注解

    使用注解须知: 1:导入约束:导入context的命名空间 2:配置注解的支持:<context:annotation-config/> <?xml version="1. ...