告别硬编码!用工厂模式优雅构建可扩展的 Spring Boot 应用 [特殊字符]
嗨,各位技术伙伴们!👋
在日常的软件开发中,我们经常面临需求变更的挑战。如何构建一个既能满足当前需求,又能轻松应对未来变化的系统呢?答案往往藏在那些经典的设计模式中。
今天,我们就来聊聊创建型模式中的明星——工厂模式 (Factory Patterns),并深入探讨如何将其与强大的 Spring Boot 框架结合,打造出专业、灵活且易于维护的企业级应用。
什么是工厂模式?🤔
简单来说,工厂模式的核心思想就是将对象的创建过程封装起来,让客户端代码与具体对象的创建细节解耦。想象一下,你不再需要亲自去 new
一个复杂的对象,而是向一个“工厂”索要,工厂会根据你的需求生产出合适的对象。
工厂模式家族主要有三位成员:
- 简单工厂 (Simple Factory):一个工厂类根据传入参数创建不同产品。简单直接,但不易扩展。
- 工厂方法 (Factory Method):定义一个创建对象的接口,但由子类决定实例化哪个类。符合开闭原则,扩展性更好。
- 抽象工厂 (Abstract Factory):用于创建一系列相关或相互依赖的对象(即一个产品族),而无需指定它们具体的类。
其中,工厂方法模式是应用非常广泛的一种。它的结构大致如下:
工厂模式在 Spring Boot 中的身影 💡
Spring 框架本身就是一个巨大的“工厂”的成功实践者!其核心的 IoC (控制反转) 容器就负责了我们应用中几乎所有对象 (Beans) 的创建、配置和管理。
在 Spring Boot 中,工厂模式的思想更是无处不在:
-
@Bean
方法:在@Configuration
类中,每一个被@Bean
注解的方法都充当了一个工厂方法。Spring 调用这些方法来创建和管理 Bean 实例。@Configuration public class AppConfig {@Bean // 我就是一个工厂方法!public MyService myService(MyDependency dependency) {return new MyServiceImpl(dependency);} }
-
FactoryBean
接口:Spring 提供的FactoryBean
接口允许我们自定义更复杂的 Bean 创建逻辑。实现了该接口的类,其getObject()
方法就是名副其实的工厂方法。 -
自动配置 (Auto-configuration):Spring Boot 的自动配置机制,在某种程度上也体现了抽象工厂的思想。它能根据不同的条件(如类路径、属性配置)为我们自动装配一整套协同工作的 Bean(一个“产品族”)。
实战演练:构建一个灵活的支付系统 💳
理论讲了不少,让我们通过一个实战项目来感受工厂模式的魅力。我们将构建一个支持多种支付方式(如支付宝、微信支付、银行卡支付)的系统。
核心思路:
- 定义一个统一的
PaymentService
支付接口。 - 为每种支付方式创建具体的实现类 (如
AlipayServiceImpl
),并将它们注册为 Spring 的@Service
Bean。 - 创建一个
PaymentServiceFactoryImpl
,它利用 Spring 的依赖注入特性,自动收集所有PaymentService
的实现。 - 提供一个方法,根据传入的支付类型,从工厂中获取相应的服务实例。
- 通过一个
PaymentController
暴露 API,接收支付请求并调用工厂进行处理。
下面是我们支付系统核心组件的简化结构图:
关键代码片段(示意):
-
PaymentService
接口:public interface PaymentService {void pay(BigDecimal amount);PaymentType getServiceType(); // 用于工厂识别 }
-
AlipayServiceImpl
(具体服务):@Service("alipayService") public class AlipayServiceImpl implements PaymentService {// ... 实现 pay() 和 getServiceType() }
-
PaymentServiceFactoryImpl
(核心工厂):@Component public class PaymentServiceFactoryImpl {private final Map<PaymentType, PaymentService> serviceCache;@Autowired // 注入所有PaymentService实现public PaymentServiceFactoryImpl(List<PaymentService> services) {// ... 初始化serviceCache}public PaymentService getPaymentService(PaymentType type) {// ... 从cache中获取} }
这种设计的最大好处是什么? 可扩展性! 当我们需要支持一种新的支付方式(比如 Apple Pay)时,只需要:
- 在
PaymentType
枚举中添加新类型。 - 创建一个
ApplePayServiceImpl
实现PaymentService
接口,并标记为@Service
。 - 完成! 工厂会自动发现并注册这个新服务,无需修改工厂或控制器的代码,完美符合开闭原则。
核心优势与总结 🌟
通过本次学习和实践,我们可以看到工厂模式(特别是结合 Spring Boot 特性)能为我们带来:
- 高度解耦:客户端与具体产品实现分离。
- 增强灵活性:轻松切换和新增产品实现。
- 提高可维护性:代码结构更清晰,职责更分明。
- 遵循开闭原则:对扩展开放,对修改关闭。
设计模式并非银弹,但它们是前人智慧的结晶,能帮助我们写出更优雅、更健壮的代码。
动手实践,一探究竟!🚀
理论与实践相结合才能真正掌握。我已经将本次学习和实践的完整代码(包括基础的工厂模式示例和 Spring Boot 支付系统项目)上传到了 GitHub,欢迎大家克隆、学习和交流!
代码仓库地址: https://github.com/Wilsoncyf/design-pattern-springboot.git
学习设计模式是一个持续精进的过程。您在项目中常用的设计模式有哪些呢?欢迎在评论区留言,一起交流探讨,共同进步!💪
相关文章:

告别硬编码!用工厂模式优雅构建可扩展的 Spring Boot 应用 [特殊字符]
嗨,各位技术伙伴们!👋 在日常的软件开发中,我们经常面临需求变更的挑战。如何构建一个既能满足当前需求,又能轻松应对未来变化的系统呢?答案往往藏在那些经典的设计模式中。 今天,我们就来聊聊…...

Express教程【006】:使用Express写接口
文章目录 8、使用Express写接口8.1 创建API路由模块8.2 编写GET接口8.3 编写POST接口 8、使用Express写接口 8.1 创建API路由模块 1️⃣新建routes/apiRouter.js路由模块: /*** 路由模块*/ // 1-导入express const express require(express); // 2-创建路由对象…...

mongodb集群之分片集群
目录 1. 适用场景2. 集群搭建如何搭建搭建实例Linux搭建实例(待定)Windows搭建实例1.资源规划2. 配置conf文件3. 按顺序启动不同角色的mongodb实例4. 初始化config、shard集群信息5. 通过router进行分片配置 1. 适用场景 数据量大影响性能 数据量大概达到千万级或亿级的时候&…...

Starrocks Full GC日志分析
GC日志样例: [2025-06-03T07:36:06.1770800] GC(227) Pause Full (G1 Evacuation Pause) [2025-06-03T07:36:06.1960800] GC(227) Phase 1: Mark live objects [2025-06-03T07:36:06.9480800] GC(227) Cleaned string and symbol table, strings: 47009 processed,…...

飞算 JavaAI 赋能老项目重构:破旧立新的高效利器
许多企业的 Java 老项目面临着代码陈旧、架构落后、维护困难等问题。老项目重构势在必行,却又因庞大的代码量、复杂的业务逻辑让开发团队望而却步。 老项目重构困境重重 传统的 Java 老项目往往在长期的迭代和维护中积累了诸多问题。一方面,代码质量堪…...
RockyLinux9安装Docker
如何在RockyLinux9下安装Docker RockyLinux采用了全新的dnf来进行包管理,dnf支持yum别名,还没习惯的可以将dnf替换为yum 确保dnf最新 sudo dnf update -y安装dnf-plugins-core包 sudo dnf install -y dnf-plugins-core yum-utils添加Docker的官方仓库…...
RequestRateLimiterGatewayFilterFactory
一、功能说明 RequestRateLimiterGatewayFilterFactory 是 Spring Cloud Gateway 的流量控制组件,用于实现 API 请求速率限制,核心功能包括: 限制单位时间内的请求数量(如每秒10次)防止服务被突发流量击垮࿰…...
解决 xmlsec.InternalError: (-1, ‘lxml xmlsec libxml2 library version mismatch‘)
解决 xmlsec.InternalError: (-1, ‘lxml & xmlsec libxml2 library version mismatch’) 错误信息如下: Traceback (most recent call last):File "/home/mobsf/Mobile-Security-Framework-MobSF/manage.py", line 18, in <module>execute_f…...
【Linux基础知识系列】第九篇-Shell脚本入门
在Linux世界中,Shell脚本是自动化任务和简化操作的重要工具。它可以帮助用户编写一系列命令,自动执行重复的任务,从而提高工作效率。在本篇文章中,我们将介绍Shell脚本的基本概念、编写方法、常用命令和结构。通过这些内容&#x…...

typescript的Interface和Type
类型别名和接口非常相似,在大多数情况下你可以在它们之间自由选择。 几乎所有的 interface 功能都可以在 type 中使用,关键区别在于不能重新开放类型以添加新的属性,而接口始终是可扩展的。 // window.ts.transpileModule(src, {}); 这是调…...

java后端生成心电图-jfreechart
用jfreechart生成心电图 先上成功的图片 上代码 1.导入包 implementation org.jfree:jfreechart:1.5.4implementation org.jfree:jcommon:1.0.242.实现代码 对数据进行滤波 转换单位 package com.shinrun.infrastructure.util;import java.util.ArrayList; import java.ut…...

算法/机理模型演示平台搭建(二)——算法接口部署(FastApi)
算法/机理模型演示平台搭建(二)—— 算法接口部署(FastApi) 1. 项目结构2. 构建 Docker 镜像3. 运行 Docker 容器4. 访问 API 文档5. 调用 API1. 项目结构 app app/algorithms app/models Dockerfile FROM python:3.9-slimWORKDIR /codeCOPY ./requirements.txt /code…...

动态规划-647.回文子串-力扣(LeetCode)
一、题目解析 这里的子字符串是连续的,与之前的子序列不同,这里需要我们统计回文子串的数目。 二、算法原理 这里也有其他算法可以解决该问题,如中心扩展算法 时间复杂度O(N^2)/空间复杂度O(1),马拉车算法(具有局限性) 时间复杂…...
es 的字段类型(text和keyword)
Text 当一个字段是要被全文检索时,比如 Email 内容、产品描述,这些字段应该使用 text 类型。设置 text 类型以后,字段内容会被分析,在生成倒排索引之前,字符串会被分析器分词。text类型的字段不用于排序,很…...
Kotlin 中companion object {} 什么时候触发
在 Kotlin 中,companion object 的初始化触发时机是一个重要但容易被忽视的细节。以下是详细的解释: 1. 基本触发时机 companion object 的初始化发生在: 首次访问该类时(无论是访问伴生对象成员、创建类实例,还是通过…...

仿真每日一练 | Workbench中接触种类及选择方法简介
Workbench中给我们提供的接触类型主要包括以下几种👇 ◆ 1、摩擦 ◆ 2、无摩擦 ◆ 3、绑定 ◆ 4、不分离 ◆ 5、粗糙 ◆ 6、强制滑移 下面通过最常用的摩擦和绑定给大家展示两者的区别,同时文末也给大家介绍了几种接触的选择方法。首先先给大家介绍一下…...

Go语言中的rune和byte类型详解
1. rune类型 1.1. 基本概念 1. rune是Go语言的内建类型,它是int32的别名,即32位有符号整数; 2. 用于表示一个Unicode码点,全拼Unicode code point; 3. 可以表示任何UTF-8编码的字符; 1.2. 特点 1. 每…...
superior哥AI系列第6期:Transformer注意力机制:AI界的“注意力革命“
🎭 superior哥AI系列第6期:Transformer注意力机制:AI界的"注意力革命" 嘿!小伙伴们!👋 今天superior哥要带你们探索AI界最火的技术——Transformer!这个家伙可了不得,它不…...

【java面试】redis篇
redis篇 一、适用场景(一)缓存1、缓存穿透1.1 解决方案1:缓存空数据,查询返回的数据为空,将空结果缓存1.2 解决方案2:布隆过滤器 2、缓存击穿1.1 解决方案1:互斥锁1.2 解决方案2:逻辑…...

高效易用的 MAC 版 SVN 客户端:macSvn 使用体验
高效易用的 MAC 版 SVN 客户端:macSvn 使用体验 下载安装使用总结 最近有个项目要使用svn, 但是mac缺乏一款像 Windows 平台 TortoiseSVN 那样全面、高效且便捷的 SVN 客户端工具, 直到博主找到了该工具本文将结合实际使用体验,详细介绍 macSvn工具的核心…...
【搭建 Transformer】
搭建 Transformer 的基本步骤 Transformer 是一种基于自注意力机制的深度学习模型,广泛应用于自然语言处理任务。以下为搭建 Transformer 的关键步骤和代码示例。 自注意力机制 自注意力机制是 Transformer 的核心,计算输入序列中每个元素与其他元素的…...
自然图像数据集
目录 CIFAR-10 数据集CIFAR-100 数据集AFHQ 数据集FFHQ 数据集 CIFAR-10 数据集 简介: CIFAR-10 是一个经典的图像分类数据集,广泛用于机器学习领域的计算机视觉算法基准测试。它包含60000幅32x32的彩色图像,分为10个类,每类6000…...
Linux下使用nmcli连接网络
Linux下使用nmcli连接网络 介绍 在使用ubuntu系统的时候,有时候不方便使用桌面,使用ssh远程连接,可能需要使用nmcli命令来连接网络。本文将介绍如何使用nmcli命令连接网络。nmcli 是 NetworkManager 的命令行工具,用于管理网络连…...

HCIP(BGP综合实验)
一、实验拓扑 AS 划分: AS1:R1(环回 L0:172.16.0.1/32,L1:192.168.1.0/24)AS2:R2、R3、R4、R5、R6、R7(内部运行 OSPF,AS 号为 64512 和 64513 的联盟)AS3:R…...

Attention Is All You Need (Transformer) 以及Transformer pytorch实现
参考https://zhuanlan.zhihu.com/p/569527564 Attention Is All You Need (Transformer) 是当今深度学习初学者必读的一篇论文。 一. Attention Is All You Need (Transformer) 论文精读 1. 知识准备 机器翻译,就是将某种语言的一段文字翻译成另一段文字。 由…...

uniapp+vue2+uView项目学习知识点记录
持续更新中... 1、发送给朋友,分享到朋友圈功能开启 利用onShareAppMessage和onShareTimeline生命周期函数,在script中与data同级去写 // 发送给朋友 onShareAppMessage() {return {title: 清清前端, // 分享标题path: /pages/index/index, // 分享路…...

精美的软件下载页面HTML源码:现代UI与动画效果的完美结合
精美的软件下载页面HTML源码:现代UI与动画效果的完美结合 在数字化产品推广中,一个设计精良的下载页面不仅能提升品牌专业度,还能显著提高用户转化率。本文介绍的精美软件下载页面HTML源码,通过现代化UI设计与丰富的动画效果&…...

车载诊断架构 --- DTC消抖参数(Trip Counter DTCConfirmLimit )
我是穿拖鞋的汉子,魔都中坚持长期主义的汽车电子工程师。 老规矩,分享一段喜欢的文字,避免自己成为高知识低文化的工程师: 做到欲望极简,了解自己的真实欲望,不受外在潮流的影响,不盲从,不跟风。把自己的精力全部用在自己。一是去掉多余,凡事找规律,基础是诚信;二是…...

javaEE->IO:
文件: 操作系统中会把很多 硬件设备 和 软件资源 抽象成“文件”,统一进行管理。 大部分谈到的文件,都是指 硬盘的文件,文件就相当于是针对“硬盘”数据的一种抽象 硬盘: 1.机械硬盘:便宜 2.固态硬盘&…...

Oracle 用户/权限/角色管理
1. 用户 1.1. 用户的创建和删除 1.1.1. 创建用户 create user user identified {by password | externally} [ default tablespace tablespace ] [ temporary tablespace tablespace ] [ quota {integer [k | m ] | unlimited } on tablespace [ quota {integer [k | m ] | …...