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

设计模式:简单工厂模式

工厂设计模式属于创建型模式,它提供了一种创建对象的最佳方式。工厂模式提供了一种创建对象的方式,而无需指定要创建的具体类。工厂模式属于创建型模式,它在创建对象时提供了一种封装机制,将实际创建对象的代码与使用代码分离。

面相对象编程中,万物皆对象。每个对象如果都通过关键字new创建,则会导致对象与其调用者耦合严重。假如更换要new的对象,则所有new对象的地方都需要改变代码,这显然违背了“开-闭原则”。如果使用工厂来生产对象,我们只需要和工厂打交道,彻底和对象解耦,如果更换对象,只需要在工厂里更换对象,而不需要再所有的地方进行更换,这样达到了与对象解耦的目的。工厂模式最大优点就是解耦

简单工厂模式

 简单工厂模式根据传入的参数,决定创建哪个类型的实例。简单工厂模式本着高内聚低耦合的原则,将系统的逻辑部分和功能分开。

使用场景 

首先当然是在你需要new一个类的对象的时候,此时各种状况出现啦:

  1. 你不想直接new这个类的对象,怕以后这个类改变的时候你需要回来改代码,而此时依赖这个类的地方已经到处都是了。
  2. 这个类的对象构建过程非常复杂,你不愿意将这么复杂的构建过程一遍又一遍的写在需要用到此对象的地方。
  3. 这个类的对象在构建过程中依赖了很多其他的类,而你无法在调用的地方提供
  • JDK源码中的 Calendar.getInstance()方法。
  • Logback源码中多个重载的 getLogger()方法。
  • QQ 换皮肤,一整套一起换。

  • 生成不同操作系统的程序。

简单工厂模式的3个角色: 

  • 对象工厂(SimpleFactory),负责实现创建所有产品实例的内部逻辑。
  • 抽象产品,即要生产的对象的接口,描述所有实例要实现的公共接口。
  • 具体产品,即创建的产品实例。所有创建的实例都是抽象产品的具体实现。

结构图:

优缺点

优点:

  • 结构简单,调用方便。工厂和产品的职责区分明确。
  • 客户端无需知道所创建具体产品的类名,只需知道参数即可。

缺点:

  • 简单工厂模式的工厂类单一,负责所有产品的创建,职责过重,一旦异常,整个系统将受影响。
  • 违背“开放-关闭原则”,一旦添加新产品就必须修改简单工厂类的内部逻辑,工厂类代码会非常臃肿,违背高聚合原则。
  • 简单工厂模式使用了静态工厂方法,静态方法不能被继承和重写,造成工厂角色无法形成基于继承的等级结构。

应用实例: 

简易计算器为例,需求:输入两个数和运算符,得到结果。

抽象产品:

public abstract class Operation {protected abstract double compute (double number1, double number2) throws Exception;
}

具体产品:

// 加法
public class OperationAdd extends Operation{@Overrideprotected double compute(double number1, double number2) {return number1 + number2;}
}// 减法
public class OperationSub extends Operation{@Overrideprotected double compute(double number1, double number2) {return number1 - number2;}
}// 乘法
public class OperationMul extends Operation{@Overrideprotected double compute(double number1, double number2) {return number1 * number2;}
}// 除法
public class OperationDiv extends Operation{@Overrideprotected double compute(double number1, double number2) throws Exception {if(number2 == 0){throw new Exception("除数不能为0!");}return number1 / number2;}
}

对象工厂:

这里对简单工厂创建产品进行了优化,可以通过缓存、配置文件、反射和泛型等技术了优化。

如果运算业务需要扩展,需要创建具体运算类即可,不需要再修改简单工厂类了。

public class OperationFactory {private static final Map<String, Operation> cachedParsers = new HashMap<>();static {cachedParsers.put("+", new OperationAdd());cachedParsers.put("-", new OperationSub());cachedParsers.put("*", new OperationMul());cachedParsers.put("/", new OperationDiv());}public static Operation createOperate0(String operate) {if (operate == null || operate.isEmpty()) {return null;//返回null还是IllegalArgumentException全凭你自己说了算}return cachedParsers.get(operate);}/***  createOperate0运用缓存技术,节省内存和对象创建的时间*  createOperate0 和 createOperate1 同理*/public static Operation createOperate1(String operate){Operation operation = null;switch (operate){case "+":operation = new OperationAdd(); break;case "-":operation = new OperationSub(); break;case "*":operation = new OperationMul(); break;case "/":operation = new OperationDiv(); break;}return operation;}/*** createOperate1方法不符合开闭原则*      如果运算业务继续扩展,需要创建具体运算类,也需要修改简单工厂的 createOperate1方法。* 采用反射技术进行优化,即createOperate2方法*/public static Operation createOperate2(String className){Operation operation = null;try {if(!(null == className || "".equals(className))){operation = (Operation) Class.forName(className).newInstance();}} catch (Exception e) {e.printStackTrace();}return operation;}/*** createOperate2方法中类型强制转型,参数为字符串,可控性不高* 采用反射和泛型技术进行优化,即createOperate3方法*/public static Operation createOperate3(Class<? extends Operation> clazz){try {if(null != clazz){return clazz.newInstance();}} catch (Exception e) {e.printStackTrace();}return null;}}

测试:

public class Test {public static void main(String[] args) throws Exception {double num1 = 10;double num2 = 2;System.out.println(OperationFactory.createOperate0("*").compute(num1, num2));System.out.println(OperationFactory.createOperate1("*").compute(num1, num2));Operation operate2 = OperationFactory.createOperate2("cn.jq.learn.simplefactorypattern.OperationDiv");System.out.println(operate2.compute(num1, num2));Operation operate3 = OperationFactory.createOperate3(OperationAdd.class);System.out.println(operate3.compute(num1, num2));}
}

相关文章:

  • 力扣72. 编辑距离
  • 系统架构设计师-22年-论文题目
  • 【Redis】list以及他的应用场景
  • PDF标准详解(一)——PDF文档结构
  • vue中,使用file-saver导出文件,下载Excel文件、下载图片、下载文本
  • C# 命名管道NamedPipeServerStream使用
  • Spring依赖注入
  • 响应式Web开发项目教程(HTML5+CSS3+Bootstrap)第2版 例5-3 getBoundingClientRect()
  • 【基础算法练习】单调队列与单调栈模板
  • LabVIEW扫频阻抗测试系统
  • 回归预测 | MATLAB实现PSO-GRNN粒子群优化广义回归神经网络多输入单输出预测(含优化前后预测可视化)
  • vue 跨域XMLHttpRequest
  • 如何使用 WebRTC 与 Kurento 建立视频会议 App
  • 如何成为一个更好的沟通者?
  • 粒子群优化算法(Particle Swarm Optimization,PSO)求解基于移动边缘计算的任务卸载与资源调度优化(提供MATLAB代码)
  • [NodeJS] 关于Buffer
  • 【React系列】如何构建React应用程序
  • 【跃迁之路】【477天】刻意练习系列236(2018.05.28)
  • 4. 路由到控制器 - Laravel从零开始教程
  • CAP 一致性协议及应用解析
  • git 常用命令
  • Javascript Math对象和Date对象常用方法详解
  • Js实现点击查看全文(类似今日头条、知乎日报效果)
  • PHP面试之三:MySQL数据库
  • UEditor初始化失败(实例已存在,但视图未渲染出来,单页化)
  • 得到一个数组中任意X个元素的所有组合 即C(n,m)
  • 猴子数据域名防封接口降低小说被封的风险
  • 码农张的Bug人生 - 初来乍到
  • 数据库写操作弃用“SELECT ... FOR UPDATE”解决方案
  • 国内开源镜像站点
  • 容器镜像
  • ​LeetCode解法汇总2696. 删除子串后的字符串最小长度
  • ​Z时代时尚SUV新宠:起亚赛图斯值不值得年轻人买?
  • $(function(){})与(function($){....})(jQuery)的区别
  • $NOIp2018$劝退记
  • (26)4.7 字符函数和字符串函数
  • (52)只出现一次的数字III
  • (ctrl.obj) : error LNK2038: 检测到“RuntimeLibrary”的不匹配项: 值“MDd_DynamicDebug”不匹配值“
  • (iPhone/iPad开发)在UIWebView中自定义菜单栏
  • (八)Flask之app.route装饰器函数的参数
  • (超详细)2-YOLOV5改进-添加SimAM注意力机制
  • (二)WCF的Binding模型
  • (附源码)ssm跨平台教学系统 毕业设计 280843
  • (学习日记)2024.04.04:UCOSIII第三十二节:计数信号量实验
  • (转载)Linux 多线程条件变量同步
  • .NET 4.0中使用内存映射文件实现进程通讯
  • .netcore 如何获取系统中所有session_ASP.NET Core如何解决分布式Session一致性问题
  • .net打印*三角形
  • .NET中使用Protobuffer 实现序列化和反序列化
  • @RequestBody的使用
  • @require_PUTNameError: name ‘require_PUT‘ is not defined 解决方法
  • [].slice.call()将类数组转化为真正的数组
  • [Android Pro] android 混淆文件project.properties和proguard-project.txt
  • [ASP.NET MVC]Ajax与CustomErrors的尴尬
  • [ASP.NET MVC]如何定制Numeric属性/字段验证消息