【创建型设计模式】工厂模式

server/2024/11/25 11:36:46/

【创建型设计模式】工厂模式

创建型设计模式第二期!本期介绍简单工厂模式和工厂方法模式。

简单工厂模式

简单工厂模式(又叫作静态工厂方法模式),其属于创建型设计模式,简单工厂模式不属于设计模式中的 23 种经典模式。提到它是为了让大家能够更好地理解后面讲到的工厂方法模式。

定义:简单工厂模式属于创建型设计模式,其又被称为静态工厂方法模式,这是由一个工厂对象决定创建出哪一种产品类的实例。

简单工厂模式通过工厂类的静态方法,根据传入的参数创建并返回不同的产品对象。

在简单工厂模式中有如下角色:

  • Factory: 工厂类,这是简单工厂模式的核心,它负责实现创建所有实例的内部逻辑。工厂类的创建产品类的方法可以被外界直接调用,创建所需的产品对象。
  • Product: 抽象产品类,这是简单工厂模式所创建的所有对象的父类,它负责描述所有实例所共有的公共接口。
  • Product: 具体产品类,这是简单工厂模式的创建目标。

省流:

工厂类(Factory):负责创建具体产品实例的类。

抽象产品(Product):所有具体产品的父类或接口。

具体产品(Concrete Product):工厂类所创建的具体对象。

1. 简单工厂模式的简单实现

这里我们用生产计算机来举例,假设有一个计算机的代工生产商,它目前已经可以代工生产联想计算机了。随着业务的拓展,这个代工生产商还要生产惠普和华硕的计算机。这样我们就需用一个单独的类来专门生产计算机,这就用到了简单工厂模式,下面我们来实现简单工厂模式。

(1)抽象产品类 (Shape 接口)
// 抽象产品
public interface Shape {void draw();
}

目的

定义统一的产品接口,所有具体产品都必须实现该接口。

面向接口编程:客户端只需要知道 Shape 的接口,而不需要关心具体实现。

优点

增强了代码的灵活性,客户端可以通过 Shape 类型调用 draw 方法,减少对具体类的依赖。

(2)具体产品类(Circle, Rectangle, Triangle 类)
// 圆形
public class Circle implements Shape {@Overridepublic void draw() {System.out.println("Drawing a Circle");}
}// 矩形
public class Rectangle implements Shape {@Overridepublic void draw() {System.out.println("Drawing a Rectangle");}
}// 三角形
public class Triangle implements Shape {@Overridepublic void draw() {System.out.println("Drawing a Triangle");}
}
目的

这些类是具体实现,负责实现抽象产品 (Shape) 中的接口方法。

每个具体产品有其独特的行为逻辑。例如,Circle.draw 可能涉及圆的特定绘制逻辑。

优点

将每个形状的行为封装在独立的类中,满足单一职责原则。

(3)工厂类(ShapeFactory)
// 工厂类
public class ShapeFactory {public static Shape createShape(String shapeType) {if ("circle".equalsIgnoreCase(shapeType)) {return new Circle();} else if ("rectangle".equalsIgnoreCase(shapeType)) {return new Rectangle();} else if ("triangle".equalsIgnoreCase(shapeType)) {return new Triangle();} else {throw new IllegalArgumentException("Unknown shape type: " + shapeType);}}
}

目的

封装对象创建逻辑:工厂类中定义了如何根据传入参数决定返回哪种 Shape 对象。

隐藏具体实现:客户端无需知道具体类名,只需通过简单的字符串参数即可获得对象实例。

实现细节

静态方法 createShape:接收一个字符串 shapeType,通过条件判断实例化对应的 Shape 对象。

错误处理:如果传入的 shapeType 无法匹配现有的类型,则抛出 IllegalArgumentException,确保程序不会意外运行。

优点

客户端与具体实现解耦,符合 依赖倒置原则

通过集中管理对象创建逻辑,可以减少冗余代码。

缺点

每增加一种产品,都需要修改 ShapeFactory 的代码,违反了 开闭原则

(4)客户端调用工厂类
public class Main {public static void main(String[] args) {// 使用工厂创建对象Shape circle = ShapeFactory.createShape("circle");circle.draw();Shape rectangle = ShapeFactory.createShape("rectangle");rectangle.draw();Shape triangle = ShapeFactory.createShape("triangle");triangle.draw();}
}
功能

客户端通过 ShapeFactory 的静态方法获取具体的 Shape 对象。

无需直接调用 new Circle()new Rectangle(),只需提供字符串参数即可完成对象的创建。

优点

客户端的代码更简洁,只需与 ShapeFactoryShape 接口交互。

隐藏了对象的具体实现,降低了耦合性。

2. 使用简单工厂模式的场景和优缺点

1. 客户端不需要关心对象创建的复杂过程

如果对象的创建涉及复杂的逻辑或依赖大量配置参数,而客户端只关心最终的产品实例,则可以使用简单工厂模式。

示例:日志记录系统根据参数返回文件日志、数据库日志、或控制台日志实例。

2. 产品种类有限且变化不频繁

当需要创建的产品种类数量较少,且不需要频繁扩展或修改时,简单工厂是一个快速实现的选择。

示例:为小型电商系统创建不同类型的支付方式(如支付宝、微信支付、信用卡支付)。

3. 需要在运行时动态决定使用哪个产品

当具体产品的选择需要根据运行时的条件决定,而这些条件可以通过参数来表示时,简单工厂模式非常适用。

示例:根据文件类型(如 .txt, .pdf)返回相应的文件解析器对象。

工厂方法模式

工厂方法模式(Factory Method Pattern)是一种 创建型设计模式,它通过定义一个抽象的工厂接口,由子类决定具体实例化的产品对象,从而让对象的创建延迟到子类中进行。

在工厂方法模式中有如下角色:

抽象工厂(Factory Interface/Abstract Factory)

  • 定义工厂的通用接口,提供一个用于创建产品的方法。

具体工厂(Concrete Factory)

  • 实现抽象工厂接口,负责具体产品的实例化。

抽象产品(Product Interface/Abstract Product)

  • 定义产品的公共接口,确保不同的产品有相同的行为。

具体产品(Concrete Product)

  • 实现抽象产品接口,提供具体的产品功能。
image-20241124003745585

工厂方法模式的优缺点

优点

符合开闭原则

  • 增加新产品时,只需添加对应的具体工厂类,而无需修改已有代码。

高内聚低耦合

  • 客户端依赖于抽象工厂和抽象产品,减少了对具体实现的耦合。

代码更灵活

  • 客户端可以通过切换具体工厂来获得不同的产品实例,而无需更改业务代码。
缺点

增加了系统复杂度

  • 每增加一种产品,就需要增加一个具体工厂类,类的数量会随着产品类型的增加而增加。

不适合产品种类较少的场景

  • 如果产品种类较少,使用工厂方法模式可能会导致代码结构过于冗长。

工厂方法模式的简单实现

工厂方法模式的抽象产品类与具体产品类的创建和简单工厂模式是一样的。

(1)创建抽象工厂
// 抽象工厂接口
public interface ParserFactory {DocumentParser createParser();
}
(2)创建具体工厂
// Word 文档解析器工厂
public class WordParserFactory implements ParserFactory {@Overridepublic DocumentParser createParser() {return new WordParser();}
}// PDF 文档解析器工厂
public class PdfParserFactory implements ParserFactory {@Overridepublic DocumentParser createParser() {return new PdfParser();}
}
(3)抽象产品
// 抽象产品接口
public interface DocumentParser {void parse(String content);
}
(4)具体产品
// Word 文档解析器
public class WordParser implements DocumentParser {@Overridepublic void parse(String content) {System.out.println("Parsing Word document: " + content);}
}// PDF 文档解析器
public class PdfParser implements DocumentParser {@Overridepublic void parse(String content) {System.out.println("Parsing PDF document: " + content);}
}
(5)客户端代码
public class Main {public static void main(String[] args) {// 使用工厂方法创建解析器ParserFactory wordFactory = new WordParserFactory();DocumentParser wordParser = wordFactory.createParser();wordParser.parse("This is a Word document");ParserFactory pdfFactory = new PdfParserFactory();DocumentParser pdfParser = pdfFactory.createParser();pdfParser.parse("This is a PDF document");}
}

工厂方法模式与简单工厂模式

特性简单工厂模式工厂方法模式
创建逻辑由单一工厂类负责,集中管理创建逻辑每个具体工厂负责创建特定类型的产品
扩展性修改工厂类即可扩展产品,但违反开闭原则添加新产品时只需添加新的工厂类,符合开闭原则
适用场景产品种类较少,变化不频繁产品种类较多,变化频繁,需要满足扩展性要求
类的复杂度工厂类单一,代码量少增加每种产品时需要增加对应的工厂类,类的数量较多
客户端与工厂的关系客户端依赖于具体的工厂类客户端依赖于抽象工厂,屏蔽具体实现细节

http://www.ppmy.cn/server/144781.html

相关文章

2024最新YT-DLP使用demo网页端渲染

2024最新YT-DLP使用demo网页端渲染 前提摘要1.使用python的fastapi库和jinjia2库进行前端渲染2.代码实现1)目录结构2)代码style.cssindex.htmlresult.htmlmain.pyrun.py 3)运行测试命令端运行 3.项目下载地址 前提摘要 2024最新python使用yt…

Epipolar-Free 3D Gaussian Splatting for Generalizable Novel View Synthesis 论文解读

目录 一、概述 二、相关工作 1、单场景3DGS 2、跨场景生成3DGS 3、几何方法解决3D任务 三、eFreeSplat 1、预训练跨视角模块 2、无外极线跨视角交互感知模块 3、迭代跨视角高斯对齐 4、高斯参数预测 一、概述 该论文设计了一种不依赖于极线约束的情况实现可推广的新视…

Spring AI Fluent API:与AI模型通信的流畅体验

引言 随着人工智能(AI)技术的飞速发展,越来越多的应用场景开始融入AI技术以提升用户体验和系统效率。在Java开发中,与AI模型通信成为了一个重要而常见的需求。为了满足这一需求,Spring AI引入了ChatClient&#xff0c…

书生浦语实战训练营L1G5000

XTuner 微调个人小助手认知任务 记录复现过程并截图。 基础任务(完成此任务即完成闯关并获得 100 算力点) 使用 XTuner 微调 InternLM2-Chat-7B 实现自己的小助手认知,如下图所示(图中的尖米需替换成自己的昵称)&…

springboot 使用笔记

1.springboot 快速启动项目 注意:该启动只是临时启动,不能关闭终端面板 cd /www/wwwroot java -jar admin.jar2.脚本启动 linux shell脚本启动springboot服务 3.java一键部署springboot 第5条 https://blog.csdn.net/qq_30272167/article/details/1…

安卓手机5G网络频繁掉4G 问题解决 手机5G网络优化方案

问题环境 在某个长期停留的位置(例如:躺平)使用手机时网络突然从5G跳到4G,偶尔跳来跳去导致网络体验很差,经过调整5G网络情况下网速及其他体验都要更好,基于这样的情况使用一种简单的操作,锁定5…

生成式AI;语义通信技术;生成式AI辅助的云边协同算法及其可解释性

目录 生成式AI 语义通信技术 生成式AI辅助的云边协同算法及其可解释性 一、端到端设计的物理层内生智能 二、生成式语义通信 三、生成式数字孪生网络 四、生成式AI辅助的云边协同算法及其可解释性 五、未来可能的研究方向 生成式AI 是一种人工智能技术,它能够从大量数…

Pytorch|mnist手写数字识别

🍨 本文为🔗365天深度学习训练营中的学习记录博客🍖 原作者:K同学啊 一、 前期准备 1. 设置GPU 如果设备上支持GPU就使用GPU,否则使用CPU import torch import torch.nn as nn import matplotlib.pyplot as plt import torchvi…