当前位置: 首页 > news >正文

【设计模式】工厂模式(定义 | 特点 | Demo入门讲解)

文章目录

    • 定义
    • 简单工厂模式
      • 案例 | 代码
        • Phone顶层接口设计
        • Meizu品牌类
        • Xiaomi品牌类
        • PhoneFactory工厂类
        • Customer 消费者类
    • 工厂方法模式
      • 案例 | 代码
        • PhoneFactory工厂类
    • Java高级特性---工厂模式与反射的高阶玩法
        • 方案:反射+工厂模式
    • 总结

其实工厂模式就是用一个代理类帮你创建管理对象,你就不用在代码层面去不断new对象的使用了。

创建对象和调用对象两者之间会解耦!

定义

工厂模式有三种:

  • 简单工厂模式
  • 工厂方法模式
  • 抽象方法模式

简单工厂模式

简单工厂模式也叫静态工厂模式。

举个例子:你要去买一台手机,你不用关心手机是怎么生产出来的,里面的零件具体又是怎么制造的,这些通通都交给工厂去处理,你尽管去买手机就好了。

image-20240705131418461

案例 | 代码

Phone顶层接口设计
/*** @author linghu* @date 2024/7/5 11:08*/
public interface Phone {void getBrand();
}
Meizu品牌类
/*** @author linghu* @date 2024/7/5 11:09*/
public class Meizu implements Phone{@Overridepublic void getBrand() {System.out.println("魅族");}
}
Xiaomi品牌类
/*** @author linghu* @date 2024/7/5 12:35*/
public class Xiaomi implements Phone{@Overridepublic void getBrand() {System.out.println("小米");}
}
PhoneFactory工厂类

这个是简单工厂模式,又叫静态工厂,所以方法自然是静态的!而且这个工厂类负责帮我们创建对象,并且返回对象!

在这里我们可以先用硬编码的方式:

/*** @author linghu* @date 2024/7/5 12:35*/
public class PhoneFactory {public static Phone getPhone(String phone){if ("小米".equals(phone)){return new Xiaomi();} else if ("魅族".equals(phone)) {return new Meizu();}else {return null;}}
}
Customer 消费者类

这个类就类似一个客户端的作用把。

/*** @author linghu* @date 2024/7/5 12:38*/
public class Customer {public static void main(String[] args) {PhoneFactory.getPhone("小米").getBrand();PhoneFactory.getPhone("魅族").getBrand();}
}

image-20240705132326792

工厂方法模式

工厂方法模式就是在上面简单工厂模式的基础上做了一些修改,具体的做法就是为每种不同品牌的手机都创建一个工厂独立生产。

image-20240705140352151

案例 | 代码

如上,其他代码都不变,变化的部分是工厂那部分!

PhoneFactory工厂类

首先修改一下PhoneFactory工厂类,将:

/*** @author linghu* @date 2024/7/5 12:35*/
public class PhoneFactory {public static Phone getPhone(String phone){if ("小米".equals(phone)){return new Xiaomi();} else if ("魅族".equals(phone)) {return new Meizu();}else {return null;}}
}

修改为:

/*** @author linghu* @date 2024/7/5 12:35*/
public interface PhoneFactory {Phone getPhone();
}

新增的两个工厂类:

/*** @author linghu* @date 2024/7/5 14:09*/
public class MeizuFactory implements PhoneFactory{@Overridepublic Phone getPhone() {return new Meizu();}
}
/*** @author linghu* @date 2024/7/5 14:08*/
public class XiaomiFactory implements PhoneFactory{@Overridepublic Phone getPhone() {return new Xiaomi();}
}

Customer 消费者类:

/*** @author linghu* @date 2024/7/5 12:38*/
public class Customer {public static void main(String[] args) {
//        PhoneFactory.getPhone("小米").getBrand();
//        PhoneFactory.getPhone("魅族").getBrand();Phone xiaomi=new XiaomiFactory().getPhone();Phone meizu=new MeizuFactory().getPhone();xiaomi.getBrand();meizu.getBrand();}
}

image-20240705142134386

Java高级特性—工厂模式与反射的高阶玩法

我们先回顾一下上面的工厂模式用法,列举一下会发现:

  • 简单工厂模式的局限性在于:如果我们扩充手机的品牌,就会违反 开闭原则
  • 工厂方法模式的局限性在于:我们扩充手机品牌需要不断new更多对象,这是耦合的最大元凶!!
方案:反射+工厂模式

PhoneFactory工厂类

/*** @author linghu* @date 2024/7/5 12:35*/
public class PhoneFactory {public static Phone getInstance(String origin) throws ClassNotFoundException, InstantiationException, IllegalAccessException {Class<?> cls = Class.forName("com.linghu.factory." + origin);Phone brand=(Phone)cls.newInstance();return brand;}
}

Customer 消费者类

这个时候我们无需再new更多对象了,也不用写死了:

/*** @author linghu* @date 2024/7/5 12:38*/
public class Customer {public static void main(String[] args) throws ClassNotFoundException, InstantiationException, IllegalAccessException {
//        PhoneFactory.getPhone("小米").getBrand();
//        PhoneFactory.getPhone("魅族").getBrand();
//        Phone xiaomi=new XiaomiFactory().getPhone();
//        Phone meizu=new MeizuFactory().getPhone();
//        xiaomi.getBrand();
//        meizu.getBrand();PhoneFactory.getInstance("Xiaomi").getBrand();}
}

image-20240705161002362

输入类名以后,反射创建对象,再通过工厂返回相关对象!!

总结

第一个静态工厂模式:在实际去开发中会发现比较常用,尽管上面讲了不符合(面向对象)OOP原则。
第二个工厂方法模式:不修改工厂类的前提,也就是说不修改已有类,实现对扩展是开发,对修改关闭。

http://www.lryc.cn/news/393767.html

相关文章:

  • Linux之计划和日志
  • C++ 多态篇
  • 【LVGL-SquareLine Studio】
  • mysqli 与mysql 区别和联系, 举例说明
  • 【SpringCloud应用框架】Nacos安装和服务提供者注册
  • 英语学习交流小程序的设计
  • 实现Java多线程中的线程间通信
  • C++模板元编程(一)——可变参数模板
  • kafka中
  • Android 获取当前电池状态
  • 【JVM 的内存模型】
  • 【雷丰阳-谷粒商城 】【分布式高级篇-微服务架构篇】【17】认证服务01—短信/邮件/异常/MD5
  • geom buffer制作
  • 微软正在放弃React
  • U盘非安全退出后的格式化危机与高效恢复策略
  • 安卓虚拟位置修改
  • 大数据面试题之Presto[Trino](5)
  • 对编程开发人员在今年的一些建议
  • VSCode设置好看清晰的字体!中文用鸿蒙,英文用Jetbrains Mono
  • SpringBoot新手快速入门系列教程四:创建第一个SringBoot的API
  • 第1集《修习止观坐禅法要》
  • markdown变量引用
  • 如何使用echart做K线图
  • Spring Boot应用使用GraalVM本地编译相关配置
  • 代码的坏味道——长函数
  • 【机器学习】基于密度的聚类算法:DBSCAN详解
  • Qt 网络编程 网络信息获取操作
  • linux中的进程以及进程管理
  • pyecharts可视化案例大全(11~20)
  • Docker在人工智能领域的应用与实战