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

设计模式:工厂方法模式 - 高扩展性与低耦合的设计之道

一、为什么需要工厂方法模式?

在软件开发中,对象创建与使用耦合会影响系统的灵活性和扩展性。以通知系统(支持邮件通知、短信通知和推送通知)为例 :直接实例化。

Notification email = new EmailNotification();
Notification sms = new SMSNotification();
Notification push = new PushNotification();

这种方式存在以下问题:

• 新增通知方式(如语音通知)时,需要修改客户端代码,违反了开闭原则。

• 客户端代码与具体通知类耦合过高,导致系统难以维护和扩展。

为了解决这些问题,引入工厂方法模式,将对象创建与使用分离,提高系统的灵活性与可扩展性。

二、什么是工厂方法模式?

工厂方法模式(Factory Method Pattern)是一种创建型设计模式,其核心思想是:定义一个创建对象的接口,将具体对象的实例化过程延迟到子类中完成。

简单来说,工厂方法模式通过引入抽象工厂,将对象的创建逻辑封装起来,客户端只需要调用工厂方法,而无需关心对象如何创建。

工厂方法模式的优势包括:高扩展性、低耦合性,以及易于维护和测试。

三、工厂方法模式的实际案例

场景:同上文提到的 通知系统(支持邮件通知、短信通知和推送通知等)。

1. 抽象通知类

定义通知的统一接口,所有具体通知类都需实现它。

public abstract class Notification {// 发送通知的抽象方法public abstract void send(String message);
}

2. 具体通知类

分别实现邮件、短信和推送通知的发送逻辑。(注:如下3个通知类的逻辑类似)

// 邮件通知类
public class EmailNotification extends Notification {@Overridepublic void send(String message) {System.out.println("Sending email with message: " + message);}
}
// 短信通知类
public class SMSNotification extends Notification {@Overridepublic void send(String message) {System.out.println("Sending SMS with message: " + message);}
}
// 推送通知类
public class PushNotification extends Notification {@Overridepublic void send(String message) {System.out.println("Sending push notification with message: " + message);}
}

3.抽象类:通知工厂基类

定义创建通知对象的工厂接口。

public abstract class NotificationFactory {// 工厂方法,创建通知对象public abstract Notification createNotification();
}

4. 具体工厂类

分别创建不同类型的通知对象。(注:如下3个通知工厂类的逻辑类似)

// 邮件通知工厂
public class EmailNotificationFactory extends NotificationFactory {@Overridepublic Notification createNotification() {return new EmailNotification();}
}
// 通知工厂
public class SMSNotificationFactory extends NotificationFactory {@Overridepublic Notification createNotification() {return new SMSNotification();}
}
// 推送通知工厂
public class PushNotificationFactory extends NotificationFactory {@Overridepublic Notification createNotification() {return new PushNotification();}
}

5.客户端代码

通过具体工厂类来创建通知对象,并调用其方法。

public class Client {public static void main(String[] args) {// 通过工厂创建邮件通知NotificationFactory emailFactory = new EmailNotificationFactory();Notification email = emailFactory.createNotification();email.send("Welcome to the system!");// 通过工厂创建短信通知NotificationFactory smsFactory = new SMSNotificationFactory();Notification sms = smsFactory.createNotification();sms.send("Your verification code is 123456.");// 通过工厂创建推送通知NotificationFactory pushFactory = new PushNotificationFactory();Notification push = pushFactory.createNotification();push.send("You have a new message.");}
}

代码结构说明

• 抽象类 Notification:定义了发送通知的抽象方法send(),所有通知类都需实现它。

• 具体通知类:EmailNotification、SMSNotification和PushNotification 都继承自Notification,并实现具体的通知行为。

• 抽象工厂类 NotificationFactory:定义了创建通知对象的工厂方法createNotification()。

• 具体工厂类:EmailNotificationFactory、SMSNotificationFactory和PushNotificationFactory 继承自Noti-ficationFactory,分别负责创建各自类型的通知对象。

• 客户端:通过具体工厂类创建通知对象,实现对象创建与使用分离。

四、工厂方法模式的价值

1. 工厂方法模式 VS 简单工厂模式

工厂方法模式是简单工厂模式的进一步抽象,通过多态性将对象创建逻辑与客户端代码解耦。客户端依赖于抽象工厂接口,而非具体工厂类,从而降低了耦合度,符合开闭原则。

对比简单工厂模式:

• 简单工厂模式:工厂类集中管理产品对象创建,新增产品时需修改工厂类代码,违反开闭原则。

• 工厂方法模式:每新增一个产品,只需扩展新的产品类和对应的工厂类,符合开闭原则,但引入了额外的工厂类,增加了类的层级和开发工作量。

简单工厂模式和工厂方法模式的共同问题:

对于新增一个产品时,客户端代码仍需修改来指定具体工厂类的问题,可通过反射机制进一步优化。
2.工厂方法模式的优势

高扩展性:新增产品时,只需增加新的产品类和工厂类,无需修改现有工厂类代码。
低耦合性:产品对象创建与业务逻辑分离,职责单一,系统更易维护。
易测试性:每个工厂类和产品类都可以独立测试,修改时影响范围小,降低测试成本。

五、总结

工厂方法模式通过引入抽象工厂和具体工厂,实现了对象创建与使用的解耦,遵循了开闭原则和单一职责原则。

它提高了系统的扩展性、可维护性和可测试性,特别适用于大型、复杂项目。在开发中,工厂方法模式能够有效管理对象创建逻辑,保持系统稳定性。

相关文章:

设计模式:工厂方法模式 - 高扩展性与低耦合的设计之道

一、为什么需要工厂方法模式? 在软件开发中,对象创建与使用耦合会影响系统的灵活性和扩展性。以通知系统(支持邮件通知、短信通知和推送通知)为例 :直接实例化。 Notification email new EmailNotification(); Noti…...

CTF web入门之命令执行 完整版

web29 文件名过滤 由于flag被过滤,需要进行文件名绕过,有以下几种方法: 1.通配符绕过 fla?.* 2.反斜杠绕过 fl\ag.php 3.双引号绕过 fl’‘ag’.php 还有特殊变量$1、内联执行等 此外 读取文件利用cat函数,输出利用system、passthru 、echo echo `nl flag.php`; ec…...

自然语言处理spaCy

spaCy 是一个流行的开源 自然语言处理(NLP) 库,专注于 高效、易用和工业化应用。它由 Explosion AI 开发,广泛应用于文本处理、信息提取、机器翻译等领域。 zh_core_web_sm 是 spaCy 提供的一个小型中文预训练语言模型&#xff0…...

在spark中,窄依赖算子map和filter会组合为一个stage,这种情况下,map和filter是在一个task内进行的吗?

在 Spark 中,当 map 和 filter 这类窄依赖(Narrow Dependency)的算子连续应用时,它们会被合并到同一个 Stage 中,并且在同一个 Task 内按顺序执行。这种优化称为 流水线(Pipeline)执行&#xff…...

MySQL切换PolarDB-X方案

一、DTS 增量同步完成后的流量切换策略 1. 切换期间的数据写入处理 • 场景:DTS 增量同步完成(Lag0)后,业务流量切换到 PolarDB-X 的瞬间可能产生 2-3 秒延迟,导致部分订单仍写入 MySQL。 • 解决方案: ◦…...

Java 开发工具:从 Eclipse 到 IntelliJ IDEA 的进化之路

Java 开发工具:从 Eclipse 到 IntelliJ IDEA 的进化之路 在 Java 开发的历史长河中,开发工具的演变不仅改变了程序员的编码方式,也深刻影响了整个行业的开发效率和代码质量。从 Eclipse 到 IntelliJ IDEA,这不仅是工具的更替&…...

GPT - 2 文本生成任务全流程

数据集下载 数据预处理 import json import pandas as pdall_data []with open("part-00018.jsonl",encoding"utf-8") as f:for line in f.readlines():data json.loads(line)all_data.append(data["text"])batch_size 10000for i in ran…...

红宝书第四十三讲:基于资料的数据可视化工具简单介绍:D3.js 与 Canvas绘图

红宝书第四十三讲:基于资料的数据可视化工具简单介绍:D3.js 与 Canvas绘图12 资料取自《JavaScript高级程序设计(第5版)》。 查看总目录:红宝书学习大纲 一、D3.js:数据驱动文档的王者 1 核心特性&#x…...

UI基础(1)

quit和close的区别: driver.close():关闭当前正在使用的窗口。 1、如果你的当前浏览器窗口只有一个情况下,它就会关闭窗口并且关闭浏览器 2、如果你的当前浏览器窗口有多个的情况下,它就会关闭driver驱动焦点所在的窗口 driver.quit():真正关闭浏览器(把所有的窗口都关闭…...

深入理解 Vue 的数据代理机制

何为数据代理? 通过一个对象代理对另一个对象中的属性的操作(读/写),就是数据代理。 要搞懂Vue数据代理这个概念,那我们就要从Object.defineProperty()入手 Object.defineProperty()是Vue中比较底层的一个方法&…...

封装,继承,多态(续)

在Java中,最基础的三原则无疑是封装,继承,多态 对于这三类,最基本同样最重要,我们是会经常遇到的,在编程中,会使用,但在考试中还有一定的不理解。 对于这点,我在这里进…...

Java excel导入/导出导致内存溢出问题,以及解决方案

excel导入/导出导致内存溢出问题,以及解决方案 1、内存溢出问题导入功能重新修正,采用SAX的流式解析数据。并结合业务流程。导出功能:由于精细化了业务流程,导致比较代码比较冗杂,就只放出最简单的案例。 1、内存溢出问…...

10 个最新 CSS 功能已在所有主流浏览器中得到支持

前言 CSS 不断发展,新功能使我们的工作更快、更简洁、更强大。得益于最新的浏览器改进(Baseline 2024),许多新功能现在可在所有主要引擎上使用。以下是您可以立即开始使用的10 CSS新功能。 1. Scrollbar-Gutter 和 Scrollbar-Co…...

思科模拟器的单臂路由,交换机,路由器,路由器只要两个端口的话,连接三台电脑该怎么办,划分VLAN,dotlq协议

单臂路由 1. 需求:让三台电脑互通 2. 在二层交换机划分vlan,并加入; 3. 将连接二层交换机和路由器的端口f0/4改为trunk模式 4. 路由器:进入连接路由器的f0/0端口将端口开启 5. 进入每个vlan设dotlq协议并设网络IP&#xff08…...

14 nginx 的 dns 缓存的流程

前言 这个是 2020年11月 记录的这个关于 nginx 的 dns 缓存的问题 docker 环境下面 前端A连到后端B 前端B连到后端A 最近从草稿箱发布这个问题的时候, 重新看了一下 发现该问题的记录中仅仅是 定位到了 nginx 这边的 dns 缓存的问题, 但是 并没有到细节, 没有到 具体的 n种…...

实战教程:使用JetBrians Rider快速部署与调试PS5和Xbox上的UE项目

面向主机游戏开发者的重大新闻!在2024.3版本中,JetBrains Rider 增加了对 PlayStation5 和 Xbox 游戏主机的支持,您可以直接在您喜欢的游戏主机上构建、部署和调试 Unreal Engine 和自定义游戏引擎。 JetBrains Rider现在支持主机游戏开发&am…...

大型语言模型中中医知识的多模态基准数据集

下载链接: https://github.com/pariskang/ZhongJing-OMNI https://github.com/pariskang/ZhongJing-OMNI 下载链接 https://github.com/pariskang/ZhongJing-OMNI.git 链接失效反馈 资源简介: ZhongJing-OMNI是第一个用于评估大型语言模型中中医知…...

专题十五:动态路由——BGP

一、BGP的基本概念 BGP(Border Gateway Protocol,边界网关协议)是一种用于在不同自治系统(AS)之间交换路由信息的外部网关协议(EGP)。通过TCP179端口建立连接。目前采用BGP4版本,IP…...

Linux命令-vim编辑

用vi或vim命令进入vim编辑器。 基础: u 撤销上一次操作。x剪切当前光标所在处的字符。yy复制当前行。dd剪切当前行。p粘贴剪贴板内容到光标下方。i切换到输入模式,在光标当前位置开始输入文本。:wq保存并退出Vim 编辑器。:q!不保存强制退出Vim 编辑器。 拓展: w光…...

hive数仓要点总结

1.OLTP和OLAP区别 OLTP(On-Line Transaction Processing)即联机事务处理,也称为面向交易的处理过程,其基本特征是前台接收的用户数据可以立即传送到计算中心进行处理,并在很短的时间内给出处理结果,是对用…...

一款安全好用的企业即时通讯平台,支持统一门户

在数字化转型的浪潮中,企业面临着信息孤岛、系统分散、协作低效等诸多挑战。BeeWorks作为一款专为企业打造的数字化底座平台,凭借其强大的企业内部应用集成能力和单点登录功能,正在成为企业数字化转型的有力推手。 数字化底座平台&#xff1…...

git安装(windows)

通过网盘分享的文件:资料(1) 链接: https://pan.baidu.com/s/1MAenYzcQ436MlKbIYQidoQ 提取码: evu6 点击next 可修改安装路径 默认就行 一般从命令行调用,所以不用创建。 用vscode,所以这么选择。...

微信小程序实战案例 - 餐馆点餐系统 阶段1 - 菜单浏览

阶段 1 – 菜单浏览(超详细版) 目标:完成「首页=菜品卡片列表」 打好 UI 地基会从 云数据库 拉取 categories / dishes 并渲染打 Git Tag v1.0‑menu 1. 技术/知识点速览 知识点关键词说明云数据库db.collection().where().…...

Dashboard的安装和基本使用

1.Dashboard简介: Dashboard是Kubernetes的Web图形用户界面(GUI),它为用户提供了一个直观的方式来管理和监控Kubernetes集群。 2.实验基础和前置条件: 本实验以Kubernetes集群环境搭建与初始化-CSDN博客为基础和前置…...

英语单词 list 11

前言 这一个 list 是一些简单的单词。感觉这个浏览单词的方法比较低效,所以准备每天最多看一个 list ,真要提升英语水平,感觉还是得直接做阅读理解题。就像我们接触中文阅读材料一样,当然光知道这个表面意思还不够,还…...

JAVA基础 - 高效管理线程隔离数据结构ThreadLocalMap

欢迎光临小站:致橡树 ThreadLocalMap 是 ThreadLocal 的核心底层数据结构,负责在每个线程中存储与 ThreadLocal 实例绑定的数据。它的设计目标是高效管理线程隔离数据,同时尽量减少内存泄漏风险。以下是其核心实现细节。 数据结构与设计目标…...

DeepSeek 接入 Word 完整教程

一、前期准备 1.1 注册并获取 API 密钥 访问 DeepSeek 平台: 打开浏览器,访问 DeepSeek 官方网站(或您使用的相应平台)。注册并登录您的账户。 创建 API 密钥: 在用户控制面板中,找到“API Keys”或“API…...

通义灵码助力Neo4J开发:快速上手与智能编码技巧

在 Web 应用开发中,Neo4J 作为一种图数据库,用于存储节点及节点间的关系。当图结构复杂化时,关系型数据库的查找效率会显著降低,甚至无法有效查找,这时 Neo4J 的优势便凸显出来。然而,由于其独特的应用场景…...

高性能文件上传服务

高性能文件上传服务 —— 您业务升级的不二选择 在当今互联网数据量激增、文件体积日益庞大的背景下,高效、稳定的文件上传方案显得尤为重要。我们的文件分块上传服务端采用业界领先的 Rust HTTP 框架 Hyperlane 开发,凭借其轻量级、低延时和高并发的特…...

p2p的发展

PCDN(P2P内容分发网络)行业目前处于快速发展阶段,面临机遇与挑战并存的局面。 一、发展机遇 技术融合推动 边缘计算与5G普及:5G的高带宽、低延迟特性与边缘计算技术结合,显著提升PCDN性能,降低延迟&#x…...