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

工厂方法模式 (Factory Method Pattern) 在Spring Boot 中的应用场景

在 Spring Boot 日常开发中,工厂方法模式(Factory Method Pattern)的应用场景非常多,它可以帮助我们优雅地创建对象,解耦对象创建逻辑,提高代码的可维护性和可扩展性。下面我将详细列举几个典型的应用场景,并提供具体的代码示例。

场景一: 创建不同类型的日志记录器 (Logger)

假设你的 Spring Boot 应用需要支持多种日志记录方式,例如:

  • 控制台日志 (Console Logger)
  • 文件日志 (File Logger)
  • 数据库日志 (Database Logger)
  • 远程日志 (Remote Logger,例如发送到 Logstash 或 Kafka)

你可以使用工厂方法模式来创建不同类型的 Logger 对象。

1. 定义 Logger 接口:

public interface Logger {void log(String message);
}

2. 创建具体的 Logger 实现类:

// 控制台日志
public class ConsoleLogger implements Logger {@Overridepublic void log(String message) {System.out.println("[Console] " + message);}
}// 文件日志
import java.io.FileWriter;
import java.io.IOException;
import java.io.PrintWriter;public class FileLogger implements Logger {private final String logFilePath;public FileLogger(String logFilePath) {this.logFilePath = logFilePath;}@Overridepublic void log(String message) {try (PrintWriter writer = new PrintWriter(new FileWriter(logFilePath, true))) {writer.println("[File] " + message);} catch (IOException e) {e.printStackTrace();}}
}// 数据库日志 (示例,省略具体数据库操作)
public class DatabaseLogger implements Logger {@Overridepublic void log(String message) {System.out.println("[Database] " + message); // 实际应用中会写入数据库// ... 数据库写入逻辑 ...}
}

3. 定义 Logger 工厂接口:

public interface LoggerFactory {Logger createLogger();
}

4. 创建具体的 Logger 工厂类:

// 控制台日志工厂
public class ConsoleLoggerFactory implements LoggerFactory {@Overridepublic Logger createLogger() {return new ConsoleLogger();}
}// 文件日志工厂
public class FileLoggerFactory implements LoggerFactory {private final String logFilePath;public FileLoggerFactory(String logFilePath) {this.logFilePath = logFilePath;}@Overridepublic Logger createLogger() {return new FileLogger(logFilePath);}
}// 数据库日志工厂
public class DatabaseLoggerFactory implements LoggerFactory {@Overridepublic Logger createLogger() {return new DatabaseLogger();}
}

5. 在 Spring Boot 中配置和使用:

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.beans.factory.annotation.Qualifier;
import org.springframework.boot.CommandLineRunner;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;@SpringBootApplication
public class LoggerFactoryExampleApplication implements CommandLineRunner {// 使用 @Qualifier 注解指定要注入的 LoggerFactory 实现@Autowired@Qualifier("fileLoggerFactory") // 或者 "consoleLoggerFactory", "databaseLoggerFactory"private LoggerFactory loggerFactory;public static void main(String[] args) {SpringApplication.run(LoggerFactoryExampleApplication.class, args);}@Overridepublic void run(String... args) throws Exception {Logger logger = loggerFactory.createLogger(); // 通过工厂创建 Loggerlogger.log("This is a log message.");}
}@Configuration
class AppConfig {// 将 LoggerFactory 的实现类注册为 Spring Bean@Beanpublic LoggerFactory fileLoggerFactory() {return new FileLoggerFactory("app.log"); // 指定日志文件路径}@Beanpublic LoggerFactory consoleLoggerFactory() {return new ConsoleLoggerFactory();}@Beanpublic LoggerFactory databaseLoggerFactory() {return new DatabaseLoggerFactory();}
}

解释:

  • 我们定义了 Logger 接口和三个具体的 Logger 实现类。
  • 我们定义了 LoggerFactory 接口和三个具体的 LoggerFactory 实现类,分别用于创建不同类型的 Logger
  • 在 Spring Boot 配置类 (AppConfig) 中,我们使用 @Bean 注解将 FileLoggerFactory 注册为 Spring Bean。
  • LoggerFactoryExampleApplication 中,我们通过 @Autowired@Qualifier 注解注入 LoggerFactory 的实例(这里注入的是 FileLoggerFactory)。
  • run 方法中,我们通过 loggerFactory.createLogger() 方法创建 Logger 对象,并调用其 log() 方法。

好处:

  • 解耦: 应用程序代码不依赖于具体的 Logger 实现类,而是依赖于 Logger 接口和 LoggerFactory 接口。
  • 可扩展: 如果要添加新的日志记录方式(例如,发送到远程日志服务器),只需要创建一个新的 Logger 实现类和一个对应的 LoggerFactory 实现类,并在 Spring Boot 配置中注册新的 LoggerFactory Bean 即可,无需修改现有代码。
  • 可配置: 可以通过配置文件或 @Qualifier 注解轻松切换不同的 Logger 实现。

场景二: 创建不同类型的支付方式 (Payment Method)

假设你的电商应用需要支持多种支付方式,例如:

  • 支付宝支付 (Alipay)
  • 微信支付 (WeChat Pay)
  • 银联支付 (UnionPay)

你可以使用工厂方法模式来创建不同类型的 Payment 对象。

1. 定义 Payment 接口:

public interface Payment {void pay(double amount);
}

2. 创建具体的 Payment 实现类:

// 支付宝支付
public class Alipay implements Payment {@Overridepublic void pay(double amount) {System.out.println("Paying " + amount + " via Alipay...");// ... 支付宝支付的具体逻辑 ...}
}// 微信支付
public class WeChatPay implements Payment {@Overridepublic void pay(double amount) {System.out.println("Paying " + amount + " via WeChat Pay...");// ... 微信支付的具体逻辑 ...}
}

3. 定义 Payment 工厂接口:

public interface PaymentFactory {Payment createPayment();
}

4. 创建具体的 Payment 工厂类:

// 支付宝支付工厂
public class AlipayFactory implements PaymentFactory {@Overridepublic Payment createPayment() {return new Alipay();}
}// 微信支付工厂
public class WeChatPayFactory implements PaymentFactory {@Overridepublic Payment createPayment() {return new WeChatPay();}
}

5. 在 Spring Boot 中配置和使用:
类似于Logger的例子,此处省略。 只需把Logger替换成Payment即可。

场景三: 创建不同类型的数据库连接 (Database Connection)

  • 定义 Connection 接口,表示数据库连接。
  • 创建不同的 Connection 实现类,例如 MySQLConnection, PostgreSQLConnection, OracleConnection
  • 定义 ConnectionFactory 接口,包含 createConnection() 方法。
  • 创建不同的 ConnectionFactory 实现类,例如 MySQLConnectionFactory, PostgreSQLConnectionFactory, OracleConnectionFactory,分别用于创建不同类型的数据库连接。
  • 在Spring Boot中使用@Configuration@Bean将具体的ConnectionFactory注册为Bean.

场景四: 创建不同类型的消息发送器 (Message Sender)

  • 定义 MessageSender 接口,表示消息发送器。
  • 创建不同的 MessageSender 实现类,例如 EmailSender, SmsSender, WeChatSender
  • 定义 MessageSenderFactory 接口,包含 createSender() 方法。
  • 创建不同的 MessageSenderFactory 实现类,例如 EmailSenderFactory, SmsSenderFactory, WeChatSenderFactory,分别用于创建不同类型的消息发送器。
  • 在Spring Boot中使用@Configuration@Bean将具体的MessageSenderFactory注册为Bean.

总结

工厂方法模式在 Spring Boot 日常开发中非常有用,它可以应用于各种需要创建不同类型对象的场景。 通过使用工厂方法模式,你可以:

  • 解耦对象的创建逻辑和使用逻辑。
  • 提高代码的可维护性和可扩展性。
  • 轻松地切换不同的对象实现。
  • 符合开闭原则。

在 Spring Boot 中,我们可以结合 Spring 的依赖注入、@Configuration@Bean 注解,更方便地使用工厂方法模式。

相关文章:

工厂方法模式 (Factory Method Pattern) 在Spring Boot 中的应用场景

在 Spring Boot 日常开发中,工厂方法模式(Factory Method Pattern)的应用场景非常多,它可以帮助我们优雅地创建对象,解耦对象创建逻辑,提高代码的可维护性和可扩展性。下面我将详细列举几个典型的应用场景&…...

linux core分析---TLS读取异常

文章目录 TLS概念core 线程调用栈查看堆栈: bt查看所有线程堆栈:core分析:锁分析代码修改:thread8 f 4 (第四层堆栈) jcallback.c:186**thread10 f4 SynStack.cpp:1175tl_send_message 加锁修改tls_table1 socket_tab加锁保护2 增加tls_table 中buse的使用3 tls_tl_read_mes…...

使用 Python paramiko 自动备份设备配置实验

一、实验拓扑: 要求:交换机 SW1 做为 SSH 服务端,桥接本地虚拟虚拟网卡;本地主机通过 python paramiko 库功能登录到 SW1 上进行配置备份;AR1 做为测试 SW1 的 SSH 客户端 二、实验环境搭建: 1、SW1 配置…...

【Python项目】文本相似度计算系统

【Python项目】文本相似度计算系统 技术简介:采用Python技术、Django技术、MYSQL数据库等实现。 系统简介:本系统基于Django进行开发,包含前端和后端两个部分。前端基于Bootstrap框架进行开发,主要包括系统首页,文本分…...

某大型业务系统技术栈介绍【应对面试】

微服务架构【图】 微服务架构【概念】 微服务架构,是一种架构模式,它提倡将单一应用程序划分成一组小的服务,服务之间互相协调、互相配合,为用户提供最终价值。在微服务架构中,服务与服务之间通信时,通常是…...

复现论文:DPStyler: Dynamic PromptStyler for Source-Free Domain Generalization

论文:[2403.16697] DPStyler: Dynamic PromptStyler for Source-Free Domain Generalization github: TYLfromSEU/DPStyler: DPStyler: Dynamic PromptStyler for Source-Free Domain Generalization 论文: 这篇论文还是在PromptStyler:Prompt-driven Style Gener…...

16-使用QtChart创建动态图表:入门指南

QtChart是Qt框架中的一个强大模块,用于创建各种类型的图表,如折线图、柱状图、饼图等。它提供了丰富的API和灵活的配置选项,使得开发者能够轻松地将数据可视化集成到应用程序中。本文将介绍如何使用QtChart创建一个简单的动态折线图&#xff…...

Python在网络安全中的应用 python与网络安全

前言 网络安全是保护网络、系统和程序免受数字攻击的做法。据估计, 2019 年该行业价值 1120 亿美元,到2021 年估计有 350 万个职位空缺。 许多编程语言用于执行与网络安全相关的日常任务,但其中一种已成为行业标准:Python&#…...

轻松搭建本地大语言模型(二)Open-WebUI安装与使用

文章目录 前置条件目标一、安装 Open-WebUI使用 Docker 部署 二、使用 Open-WebUI(一)访问Open-WebUI(二)注册账号(三)模型选择(四)交互 四、常见问题(一)容器…...

解锁机器学习核心算法 | 随机森林算法:机器学习的超强武器

一、引言 在机器学习的广阔领域中,算法的选择犹如为一场冒险挑选趁手的武器,至关重要。面对海量的数据和复杂的任务,合适的算法能够化繁为简,精准地挖掘出数据背后隐藏的模式与价值。机器学习领域有十大核心算法,而随…...

Linux环境Docker使用代理推拉镜像

闲扯几句 不知不觉已经2月中了,1个半月忙得没写博客,这篇其实很早就想写了(可追溯到Docker刚刚无法拉镜像的时候),由于工作和生活上的事比较多又在备考软考架构,拖了好久…… 简单记录下怎么做的&#xf…...

构建高效 Python Web 应用:框架与服务器的选择及实践

构建高效 Python Web 应用:框架与服务器的选择及实践 flyfish 从选择合适的 Web 框架(如 Flask 和 FastAPI)到部署时选用适当的 Web 服务器(如 Waitress、Gunicorn、uWSGI 和 Uvicorn)的全过程。它不仅介绍了各个框架…...

深度学习05 ResNet残差网络

目录 传统卷积神经网络存在的问题 如何解决 批量归一化BatchNormalization, BN 残差连接方式 ​残差结构 ResNet网络 ResNet 网络是在 2015年 由微软实验室中的何凯明等几位大神提出,斩获当年ImageNet竞赛中分类任务第一名,目标检测第一名。获得CO…...

Java零基础入门笔记:(4)方法

前言 本笔记是学习狂神的java教程,建议配合视频,学习体验更佳。 【狂神说Java】Java零基础学习视频通俗易懂_哔哩哔哩_bilibili 第1-2章:Java零基础入门笔记:(1-2)入门(简介、基础知识)-CSDN博客 第3章…...

DeepSeek 和 ChatGPT 在特定任务中的表现:逻辑推理与创意生成

🎁个人主页:我们的五年 🔍系列专栏:Linux网络编程 🌷追光的人,终会万丈光芒 🎉欢迎大家点赞👍评论📝收藏⭐文章 ​ Linux网络编程笔记: https://blog.cs…...

BS5852英国家具防火安全条款主要包括哪几个方面呢?

什么是BS5852检测? BS5852是英国针对家用家具的强制性安全要求,主要测试家具在受到燃烧香烟和火柴等火源时的可燃性。这个标准通常分为四个部分进行测试,但实际应用中主要测试第一部分和第二部分,包括烟头测试和利用乙炔火焰模拟…...

VSCode本地python包“无法解析导入”

问题现象 在使用 VSCode 编写 Python 代码时,虽然程序能正常运行,但遇到“无法解析导入”的问题,导致代码无法高亮。 解决方法 配置 python.autoComplete.extraPaths 打开 VSCode 设置(CtrlShiftP -> Preferences: Open Wo…...

本地使用docker部署DeepSeek大模型

1、相关技术介绍 1.1、RAG RAG(Retrieval Augmented Generation),即“检索,增强,生成”,用于提升自然语言处理任务的性能。其核心思想是通过检索相关信息来增强生成模型的能力,具体步骤如下&am…...

Java 不可变集合

1.不可变集合 在 Java 中,不可变集合(Immutable Collections)是指在创建之后无法修改的集合。这些集合不允许添加、删除或修改元素,一旦创建后,内容就不能改变。Java 提供了一些内置的方式来创建不可变集合&#xff0…...

【全栈】SprintBoot+vue3迷你商城-细节解析(1):Token、Jwt令牌、Redis、ThreadLocal变量

【全栈】SprintBootvue3迷你商城-细节解析(1):Token、Jwt令牌、Redis、ThreadLocal变量 往期的文章都在这里啦,大家有兴趣可以看一下 后端部分: 【全栈】SprintBootvue3迷你商城(1) 【全栈】…...

统计5分钟nginx访问日志的数据并设置阈值告警

1.脚本统计生成deny_ip文件 #!/usr/bin/bash#Tate:2022.3.30 #Author:Yingjian #function: 统计5分钟之内的访问ip #env#脚本存放的目录 workdircd $(dirname $0);pwdif [ $# -eq 0 ];then echo "Usage: $0 {统计几分钟内的ip}" exit 2 filogfile/var/log/nginx/acc…...

用什么办法能实现ubuntu里面运行的自己开发的python程序能自动升级。

要实现Ubuntu中自己开发的Python程序自动升级,可以通过以下几种方式: 1. 使用 Git 仓库 定时任务 如果你的Python程序托管在Git仓库中,可以通过定时拉取最新代码来实现自动升级。 步骤: 确保Python程序在Git仓库中。在Ubuntu上…...

day12_调度和可视化

文章目录 day12_调度和可视化一、任务调度1、开启进程2、登入UI界面3、配置租户4、创建项目5、创建工作流5.1 HiveSQL部署(掌握)5.2 SparkDSL部署(掌握)5.3 SparkSQL部署(熟悉)5.4 SeaTunnel部署&#xff0…...

力扣第4题 寻找两个正序数组的中位数

力扣第4题 给定两个大小分别为 m 和 n 的正序(从小到大)数组 nums1 和 nums2。请你找出并返回这两个正序数组的 中位数 。 算法的时间复杂度应该为 O(log (mn)) 。 示例 1: 输入:nums1 [1,3], nums2 [2] 输出:2.0000…...

DC-6靶机渗透测试全过程

目录 前期准备 一、渗透测试 1.IP地址查询 2.端口信息搜寻 3.网页信息搜集 wappalyzer WPScan 反弹shell graham用户 反弹出jens的shell nmap提权 二、总结 前期准备 攻击机: kali windows11 靶机:DC-6靶机(调至NAT模式&#xff0…...

Win11 远程 连接 Ubuntu20.04(局域网)

Win11 远程 连接 Ubuntu20.04(局域网) 0. Ubuntu 开启共享1. Ubuntu系统中安装RDP服务器2.windows中连接使用方式1:远程桌面连接(winr: mstsc)方式2:mobaXterm 3 问题远程连接后出现黑屏 参考文献: 0. Ubuntu 开启共享 在ubunt设置中&#x…...

探索Hugging Face:开源AI社区的核心工具与应用实践

引言:AI民主化的先锋 在自然语言处理(NLP)领域,Hugging Face已成为开源社区的代名词。这个成立于2016年的平台,通过提供易用的工具和丰富的预训练模型库,彻底改变了开发者使用和部署AI模型的方式。截至202…...

文件分片上传 python

服务端功能 上传分片保存 app.route(/upload_filesliceprocess, methods[POST]) def upload_filesliceprocess(): file request.files[file] name_index request.form[name_index] complete request.form[complete] process request.form[process] c…...

外汇掉期(FX Swap):全球企业管理外汇风险的关键工具(中英双语)

外汇掉期(FX Swap):全球企业管理外汇风险的关键工具 引言 在全球化经济环境下,跨国公司、银行和金融机构经常面临外汇风险,因为它们的业务涉及多种货币。例如,一家中国公司可能需要欧元支付欧洲供应商&am…...

Visual Studio Code支持WSL,直接修改linux/ubuntu中的文件

步骤1 开始通过 WSL 使用 VS Code | Microsoft Learn 点击远程开发扩展包。 步骤2 Remote Development - Visual Studio Marketplace 点击install, 允许打开Visual Studio Code。 步骤3 共有4项,一齐安装。 步骤4 在WSL Linux(Ubuntu)中&#xf…...