一文彻底搞明白工厂方法模式 工厂方法模式符合什么原则
yuyutoo 2024-10-15 17:01 11 浏览 0 评论
本篇讲解Java设计模式中的工厂方法模式,分为定义、模式应用前案例、结构、模式应用后案例、适用场景、模式可能存在的困惑和本质探讨7个部分。
定义
工厂方法模式定义了一个创建对象的接口,但由子类决定要实例化的类是哪一个。工厂方法让类把实例化推迟到子类。
简单工厂在实际应用中也经常用到,它是工厂方法的一种特例。
在新的分类方式中,工厂方法模式被划分至类之间的交互类别中,其简化的是调用方与对象创建方之间的交互
模式应用前案例
在工厂方法模式以及后面的抽象工厂模式中,我们选择一个不同格式文件导出的案例。
先来看一下未使用工厂方法模式之前的代码,如下所示。
声明一个文件导出的接口,以及分别定义PDF和CSV两个具体实现的类。
public interface IExporter {//声明一个文件导出接口
void export(String data);
}
public class CSVExporter implements IExporter {//CSV文件格式导出
@Override
public void export(String data) {
System.out.println("Exporting data to CSV: " + data);
}
}
public class PDFExporter implements IExporter {//PDF文件格式导出
@Override
public void export(String data) {
System.out.println("Exporting data to PDF: " + data);
}
}
调用方代码如下:
public class Client {//调用方代码
private static IExporter getExporter(String type){
if (type.equals("PDF")) {
return new PDFExporter();
} else if (type.equals("CSV")) {
return new CSVExporter();
} else {
throw new IllegalArgumentException("Invalid exporter type.");
}
}
public static void main(String[] args) {
IExporter pdfExporter = getExporter("PDF");
pdfExporter.export("Sample PDF Data");
IExporter csvExporter = getExporter("CSV");
csvExporter.export ("sample Csv Data");
}
}
上述代码的主要问题在于调用方与每一种格式的具体实现类是紧耦合关系。实现类如果有变化或扩展,调用方都可能需要跟着变更,违背OCP开闭原则。
结构
对于工厂方法模式来说,由简到难又可以分成简单工厂模式、工厂方法模式和工厂中的工厂三种方式。
1.简单工厂模式
简单工厂模式是工厂方法模式的一种特例。在该模式中,工厂类使用一个简单的类,未使用接口实现或继承等家族方式。
简单工厂模式的主要问题在于SimpleFactory类中需要维护分支判断逻辑,如果后续Product实现类需要扩展,那么SimpleFactory类需要一同扩展,有些违背“对扩展开放、对修改关闭”的OCP原则。后面的工厂方法模式则主要为了解决该问题。
简单工厂的示例代码实现如下:
public interface Product {
void func();
}
public class ConcreteProductA implements Product {
@Override
public void func() {
System.out.println("Product A is funcing...");
}
}
public class ConcreteProductB implements Product {
@Override
public void func() {
System.out.println("Product B is funcing...");
}
}
public class SimpleFactory {
public Product createProduct(String type) {
if ("A".equalsIgnoreCase(type)) {
return new ConcreteProductA();
} else if ("B".equalsIgnoreCase(type)) {
return new ConcreteProductB();
}
return null;
}
}
public class Client {
public static void main(String[] args) {
SimpleFactory factory = new SimpleFactory();
Product p1 = factory.createProduct("A");
Product p2 = factory.createProduct("B");
p1.func();
p2.func();
}
}
2.工厂方法模式
为了解决简单工厂模式中的违背OCP原则的问题,工厂方式模式将SimpleFactory工厂类扩展成一个工厂类家族(可使用接口或继承实现),这样分支判断逻辑可以不用在工厂类中实现。
但是,这样也带来了后面的问题,即调用方需要知晓工厂类家族中的每一个具体实现类,两者是紧耦合关系。
此外,后续扩展一种文件格式,工厂实现类也需要一并扩展,违背OCP开闭原则。
工厂方法模式的示例代码实现如下:
public interface Product {
void func();
}
public class ConcreteProductA implements Product {
@Override
public void func() {
System.out.println("Product A is funcing...");
}
}
public class ConcreteProductB implements Product {
@Override
public void func() {
System.out.println("Product B is funcing...");
}
}
public abstract class Factory {
public abstract Product factoryMethod();
public void doSomething(){
Product p = factoryMethod();
p.func();
}
}
public class ConcreteFactoryA extends Factory {
@Override
public Product factoryMethod() {
return new ConcreteProductA();
}
}
public class ConcreteFactoryB extends Factory {
@Override
public Product factoryMethod() {
return new ConcreteProductA();
}
}
public class Client {
public static void main(String[] args) {
//使用 ConcreteFactory创建 Factory对象
Factory fa = new ConcreteFactoryA();
Product pa = fa.factoryMethod();
pa.func();
Factory fb = new ConcreteFactoryB();
Product pb = fa.factoryMethod();
pb.func();
}
}
3.工厂方法+简单工厂模式(工厂中的工厂)
为了解决工厂方式模式中的调用方和工厂家族类紧耦合的问题,可以前面再增加一个简单工厂ExporterFactoryMap类,由它屏蔽调用方与具体工厂实现类之间的交互问题。
模式应用后案例
接下来继续上面的文件导出案例,分别使用简单工厂模式、工厂方式模式和工厂中的工厂来进行实现。
1.简单工厂模式
声明一个文件导出的接口,以及分别定义PDF和CSV两个具体实现的类。
public interface IExporter {//文件导出接口
void export(String data);
}
public class CSVExporter implements IExporter {//CSV格式文件导出
@Override
public void export(String data) {
System.out.println("Exporting data to CSV: " + data);
}
}
public class PDFExporter implements IExporter {//PDF文件格式导出
@Override
public void export(String data) {
System.out.println("Exporting data to PDF: " + data);
}
}
然后,定义一个简单工厂类。可以发现,该工厂类中含有具体文件类的分支判断逻辑。缺点在于如果后续文件导出文件有扩展,工厂类一并扩展。
public class ExporterFactory {//简单工厂SimpleFactory
public static IExporter createExporter(String type) {
if (type.equals("PDF")) {
return new PDFExporter();
} else if (type.equals("CSV")) {
return new CSVExporter();
} else {
throw new IllegalArgumentException("Invalid exporter type.");
}
}
}
调用方代码如下。
public class Client {//简单工厂SimpleFactory的应用
private static IExporter getExporter(String type){
if (type.equals("PDF")) {
return ExporterFactory.createExporter("PDF");
} else if (type.equals("CSV")) {
return ExporterFactory.createExporter ("CSV");
} else {
throw new IllegalArgumentException("Invalid exporter type.");
}
}
public static void main(String[] args) {
// 使用简单工厂创建并使用不同类型的导出对象
IExporter pdfExporter = getExporter("PDF");
pdfExporter.export("Sample PDF Data");
IExporter csvExporter = getExporter("CSV");
csvExporter.export ("sample Csv Data") ;
}
}
2.工厂方法模式
声明一个文件导出的接口,以及分别定义PDF和CSV两个具体实现的类。
public interface IExporter {//文件导出接口
void export(String data);
}
public class CSVExporter implements IExporter {//CSV文件格式导出
@Override
public void export(String data) {
System.out.println("Exporting data to CSV: " + data);
}
}
public class PDFExporter implements IExporter {//PDF文件格式导出
@Override
public void export(String data) {
System.out.println("Exporting data to PDF: " + data);
}
}
然后,由简单工厂模式中的一个类扩展成一个家族,该案例中具体包括一个工厂接口以及CSV和PDF两个文件格式的工厂类。
public interface IExporterFactory {//工厂接口
// 工厂创建方法
public IExporter createExporter();
}
public class CSVExporterFactory implements IExporterFactory {//CSV文件格式工厂
@Override
public IExporter createExporter() {
return new CSVExporter();
}
}
class PdfExporterFactory implements IExporterFactory {//PDF文件格式工厂
@Override
public IExporter createExporter() {
return new PDFExporter();
}
}
工厂方法的调用方代码如下。可以发现,原来简单工厂模式中的分支判断逻辑已经移动到调用方代码中实现。这样一来,调用方和工厂的接口和实现类都是紧耦合关系。
public class Client {//工厂方法的调用方
private static IExporterFactory getExporterFactory(String type) {
if (type.equals("PDF")) {
return new PdfExporterFactory();
} else if (type.equals("CSV")) {
return new CSVExporterFactory();
} else {
throw new IllegalArgumentException("Invalid exporter type.");
}
}
public static void main(String[] args){
String type = "PDF";
IExporterFactory factory = getExporterFactory(type);
IExporter exporter = factory.createExporter();
String data = "test data";
exporter.export(data);
}
}
3.工厂方法+简单工厂模式(工厂中的工厂)
为了解决工厂方式模式中的上述紧耦合问题,再增加一个简单工厂,代码如下所示。
public class ExporterFactoryMap {//工厂类的简单工厂
private static final Map<String, IExporterFactory> cachedFactories = new HashMap<String, IExporterFactory>();
static {
cachedFactories.put("PDF", new PdfExporterFactory());
cachedFactories.put("CSV", new CSVExporterFactory());
}
public static IExporterFactory getExporterFactory(String type){
return cachedFactories.get(type.toUpperCase());
}
}
客户端代码修改如下。可以看出,现在客户端不需要知晓具体的工厂类,只需要知晓ExporterFactoryMap这个简单工厂类即可。
public class Client {//工厂方法模式+简单工厂=工厂的工厂
public static void main(String[] args){
String type = "PDF";
IExporterFactory factory = ExporterFactoryMap.getExporterFactory(type);
IExporter exporter = factory.createExporter();
String data = "test data2";
exporter.export(data);
}
}
适用场景
工厂方法模式的适用场景包括:
1、代码中存在条件分支的情形下,比如根据不同文件格式、不同支付方式等
2、存在创建家族类的情形下,比如接口类和多个实现类、抽象类和多个实现类、父类和多个子类的场景下
模式可能存在的困惑
困惑1:简单工厂模式和工厂方法+简单工厂模式相比,两者最后通过简单工厂类封装了分支判断逻辑,好像并没有什么优势。
因为上面的示例代码比较简单。如果创建对象的逻辑比较复杂,而且实现类又比较多。如果这些逻辑都放在简单工厂模式中的SimpleFactory类中,这个类的职责过多,后续维护困难。
相比较而言,工厂方法+简单工厂模式,将复杂对象的创建逻辑分散到不同的工厂实现类中,这样每个类职责更加单一。
困惑2:不论哪种方法,好像都没有完全消除分支判断逻辑,不同的模式仅是将其转移到不同的地方?
确实是这样。分支逻辑不可能完全消除。但是,有一点需要格外注意,分支逻辑尽可能不要放在核心类中,而是要放到支撑类当中。核心类要尽可能保持稳定性。
困惑3:编程中的OCP开闭原则,对扩展开放和对修改封闭,放到工厂方法模式中的任一模式中,好像都不符合要求。
简单工厂模式中,如果文件格式扩展,简单工厂类需要一并扩展;在工厂方法模式中,如果文件格式扩展,工厂实现类需要一并扩展;在工厂方法+简单工厂模式中,如果文件格式扩展,工厂实现类和简单工厂类都要一并扩展。严格按照OCP原则来说,都不符合要求。
实际上,完全做到OCP原则几乎是不可能的。即便使用反射+配置文件的方式。不同的文件格式放在配置文件中实现,后续配置文件也需要扩展。
OCP原则更多讲的是如何以最小的成本实现扩展。具体到设计模式中,大家要遵循在扩展时核心类不要修改,支撑类可以修改的原则。
本质
在未使用工厂模式之前,传统创建对象都是采用new方式,这种方式的主要缺点在于调用方和被调用方两者直接依赖,耦合性强。
工厂模式的本质也是要为调用方创建对象。那么它最优先实现的一点即是要实现与调用方交互尽可能简单的情况下,满足调用方获取对象的需求。
在这个要求下,简单工厂模式和工厂方法+简单工厂模式,对调用方暴露的信息是差不多一致的。选择哪种方式最终取决于创建对象的逻辑是否复杂。
相关推荐
- Mysql和Oracle实现序列自增(oracle创建序列的sql)
-
Mysql和Oracle实现序列自增/*ORACLE设置自增序列oracle本身不支持如mysql的AUTO_INCREMENT自增方式,我们可以用序列加触发器的形式实现,假如有一个表T_WORKM...
- 关于Oracle数据库12c 新特性总结(oracle数据库19c与12c)
-
概述今天主要简单介绍一下Oracle12c的一些新特性,仅供参考。参考:http://docs.oracle.com/database/121/NEWFT/chapter12102.htm#NEWFT...
- MySQL CREATE TABLE 简单设计模板交流
-
推荐用MySQL8.0(2018/4/19发布,开发者说同比5.7快2倍)或同类型以上版本....
- mysql学习9:创建数据库(mysql5.5创建数据库)
-
前言:我也是在学习过程中,不对的地方请谅解showdatabases;#查看数据库表createdatabasename...
- MySQL面试题-CREATE TABLE AS 与CREATE TABLE LIKE的区别
-
执行"CREATETABLE新表ASSELECT*FROM原表;"后,新表与原表的字段一致,但主键、索引不会复制到新表,会把原表的表记录复制到新表。...
- Nike Dunk High Volt 和 Bright Spruce 预计将于 12 月推出
-
在街上看到的PandaDunk的超载可能让一些球鞋迷们望而却步,但Dunk的浪潮仍然强劲,看不到尽头。我们看到的很多版本都是为女性和儿童制作的,这种新配色为后者引入了一种令人耳目一新的新选择,而...
- 美国多功能舰载雷达及美国海军舰载多功能雷达系统技术介绍
-
多功能雷达AN/SPY-1的特性和技术能力,该雷达已经在美国海军服役了30多年,其修改-AN/SPY-1A、AN/SPY-1B(V)、AN/SPY-1D、AN/SPY-1D(V),以及雷神...
- 汽车音响怎么玩,安装技术知识(汽车音响怎么玩,安装技术知识视频)
-
全面分析汽车音响使用或安装技术常识一:主机是大多数人最熟习的音响器材,有关主机的各种性能及规格,也是耳熟能详的事,以下是一些在使用或安装时,比较需要注意的事项:LOUDNESS:几年前的主机,此按...
- 【推荐】ProAc Response系列扬声器逐个看
-
有考牌(公认好声音)扬声器之称ProAcTablette小音箱,相信不少音响发烧友都曾经,或者现在依然持有,正当大家逐渐掌握Tablette的摆位设定与器材配搭之后,下一步就会考虑升级至表现更全...
- #本站首晒# 漂洋过海来看你 — BLACK&DECKER 百得 BDH2000L无绳吸尘器 开箱
-
作者:初吻给了烟sco混迹张大妈时日不短了,手没少剁。家里有了汪星人,吸尘器使用频率相当高,偶尔零星打扫用卧式的实在麻烦(汪星人:你这分明是找借口,我掉毛是满屋子都有,铲屎君都是用卧式满屋子吸的,你...
- 专题|一个品牌一件产品(英国篇)之Quested(罗杰之声)
-
Quested(罗杰之声)代表产品:Q212FS品牌介绍Quested(罗杰之声)是录音监听领域的传奇品牌,由英国录音师RogerQuested于1985年创立。在成立Quested之前,Roger...
- 常用半导体中英对照表(建议收藏)(半导体英文术语)
-
作为一个源自国外的技术,半导体产业涉及许多英文术语。加之从业者很多都有海外经历或习惯于用英文表达相关技术和工艺节点,这就导致许多英文术语翻译成中文后,仍有不少人照应不上或不知如何翻译。为此,我们整理了...
- Fyne Audio F502SP 2.5音路低音反射式落地音箱评测
-
FyneAudio的F500系列,有新成员了!不过,新成员不是新的款式,却是根据原有款式提出特别版。特别版产品在原有型号后标注了SP字样,意思是SpecialProduction。Fyne一共推出...
- 有哪些免费的内存数据库(In-Memory Database)
-
以下是一些常见的免费的内存数据库:1.Redis:Redis是一个开源的内存数据库,它支持多种数据结构,如字符串、哈希表、列表、集合和有序集合。Redis提供了快速的读写操作,并且支持持久化数据到磁...
- RazorSQL Mac版(SQL数据库查询工具)
-
RazorSQLMac特别版是一款看似简单实则功能非常出色的SQL数据库查询、编辑、浏览和管理工具。RazorSQLformac特别版可以帮你管理多个数据库,支持主流的30多种数据库,包括Ca...
你 发表评论:
欢迎- 一周热门
-
-
前端面试:iframe 的优缺点? iframe有那些缺点
-
带斜线的表头制作好了,如何填充内容?这几种方法你更喜欢哪个?
-
漫学笔记之PHP.ini常用的配置信息
-
其实模版网站在开发工作中很重要,推荐几个参考站给大家
-
推荐7个模板代码和其他游戏源码下载的网址
-
[干货] JAVA - JVM - 2 内存两分 [干货]+java+-+jvm+-+2+内存两分吗
-
正在学习使用python搭建自动化测试框架?这个系统包你可能会用到
-
织梦(Dedecms)建站教程 织梦建站详细步骤
-
【开源分享】2024PHP在线客服系统源码(搭建教程+终身使用)
-
2024PHP在线客服系统源码+完全开源 带详细搭建教程
-
- 最近发表
-
- Mysql和Oracle实现序列自增(oracle创建序列的sql)
- 关于Oracle数据库12c 新特性总结(oracle数据库19c与12c)
- MySQL CREATE TABLE 简单设计模板交流
- mysql学习9:创建数据库(mysql5.5创建数据库)
- MySQL面试题-CREATE TABLE AS 与CREATE TABLE LIKE的区别
- Nike Dunk High Volt 和 Bright Spruce 预计将于 12 月推出
- 美国多功能舰载雷达及美国海军舰载多功能雷达系统技术介绍
- 汽车音响怎么玩,安装技术知识(汽车音响怎么玩,安装技术知识视频)
- 【推荐】ProAc Response系列扬声器逐个看
- #本站首晒# 漂洋过海来看你 — BLACK&DECKER 百得 BDH2000L无绳吸尘器 开箱
- 标签列表
-
- mybatis plus (70)
- scheduledtask (71)
- css滚动条 (60)
- java学生成绩管理系统 (59)
- 结构体数组 (69)
- databasemetadata (64)
- javastatic (68)
- jsp实用教程 (53)
- fontawesome (57)
- widget开发 (57)
- vb net教程 (62)
- hibernate 教程 (63)
- case语句 (57)
- svn连接 (74)
- directoryindex (69)
- session timeout (58)
- textbox换行 (67)
- extension_dir (64)
- linearlayout (58)
- vba高级教程 (75)
- iframe用法 (58)
- sqlparameter (59)
- trim函数 (59)
- flex布局 (63)
- contextloaderlistener (56)