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

Java抽象工厂模式详解

Java 抽象工厂模式是一种创建型设计模式,它提供了一种方式,可以将一组具有共同主题的单个工厂封装起来,而不必指定它们具体的类。这种模式属于创建型模式,它隐藏了对象创建的逻辑,将客户端与具体类的实现解耦。

一、核心概念

1. 抽象工厂(Abstract Factory)
  • 定义:声明创建抽象产品的方法的接口或抽象类。
  • 作用:为创建一组相关或依赖的对象提供一个接口,而无需指定它们的具体类。
2. 具体工厂(Concrete Factory)
  • 定义:实现抽象工厂接口,负责创建具体产品的实例。
  • 作用:封装了一组产品的创建逻辑,客户端通过调用具体工厂的方法来获取产品对象。
3. 抽象产品(Abstract Product)
  • 定义:声明具体产品必须实现的接口或抽象类。
  • 作用:定义了产品的公共接口,所有具体产品都必须实现这些接口。
4. 具体产品(Concrete Product)
  • 定义:实现抽象产品接口的具体类。
  • 作用:由具体工厂创建,它们的实例在客户端代码中被使用。

二、示例实现

假设我们要构建一个跨平台的 UI 组件库,支持 Windows 和 MacOS 系统,每个系统都有自己的按钮和文本框实现。

1. 定义抽象产品接口
// 抽象按钮接口
interface Button {void paint();
}// 抽象文本框接口
interface TextField {void render();
}
2. 实现具体产品类
// Windows系统的按钮实现
class WindowsButton implements Button {@Overridepublic void paint() {System.out.println("绘制Windows风格按钮");}
}// Windows系统的文本框实现
class WindowsTextField implements TextField {@Overridepublic void render() {System.out.println("渲染Windows风格文本框");}
}// MacOS系统的按钮实现
class MacOSButton implements Button {@Overridepublic void paint() {System.out.println("绘制MacOS风格按钮");}
}// MacOS系统的文本框实现
class MacOSTextField implements TextField {@Overridepublic void render() {System.out.println("渲染MacOS风格文本框");}
}
3. 定义抽象工厂接口
interface GUIFactory {Button createButton();TextField createTextField();
}
4. 实现具体工厂类
// Windows系统的工厂实现
class WindowsFactory implements GUIFactory {@Overridepublic Button createButton() {return new WindowsButton();}@Overridepublic TextField createTextField() {return new WindowsTextField();}
}// MacOS系统的工厂实现
class MacOSFactory implements GUIFactory {@Overridepublic Button createButton() {return new MacOSButton();}@Overridepublic TextField createTextField() {return new MacOSTextField();}
}
5. 客户端代码
class Application {private Button button;private TextField textField;public Application(GUIFactory factory) {this.button = factory.createButton();this.textField = factory.createTextField();}public void paint() {button.paint();textField.render();}
}
6. 使用示例
public class Main {public static void main(String[] args) {// 根据当前操作系统创建对应的工厂GUIFactory factory;String osName = System.getProperty("os.name").toLowerCase();if (osName.contains("windows")) {factory = new WindowsFactory();} else {factory = new MacOSFactory();}// 创建应用程序并使用工厂创建UI组件Application app = new Application(factory);app.paint();}
}

三、抽象工厂模式的优缺点

优点
  1. 解耦客户端与具体实现:客户端通过抽象接口操作对象,无需依赖具体类。
  2. 便于切换产品系列:只需更换具体工厂,即可在不同产品系列间切换。
  3. 保证产品一致性:具体工厂创建的产品是相互兼容的。
  4. 符合开闭原则:新增产品系列时,无需修改现有代码,只需添加新的工厂和产品类。
缺点
  1. 扩展困难:如果需要增加新的产品类型,需要修改抽象工厂接口及其所有实现类,违反开闭原则。
  2. 代码复杂度增加:随着产品系列的增加,类的数量会显著增加。

四、适用场景

  1. 系统不依赖产品创建和表示的细节:客户端只需知道产品接口,无需关心实现。
  2. 系统需要使用多个产品系列中的一个:如跨平台应用、游戏中的不同主题等。
  3. 产品系列相互关联或依赖:例如,一套完整的 UI 组件(按钮、文本框、下拉菜单等)需要保持风格一致。

五、与工厂方法模式的对比

维度工厂方法模式抽象工厂模式
核心单个产品的创建逻辑封装多个相关产品的创建逻辑封装
工厂类结构一个抽象工厂类,多个具体工厂类一个抽象工厂接口,多个具体工厂类
产品类型一种产品类型多种相关产品类型
扩展性新增产品时更灵活新增产品系列时更灵活

六、在 Java 中的实际应用

  1. Java 的 AWT/Swing 库:根据当前操作系统创建不同风格的 UI 组件。
  2. 数据库连接:使用不同的数据库驱动(MySQL、Oracle 等)时,通过抽象工厂创建相应的连接对象。
  3. 游戏开发:创建不同主题的游戏元素(如奇幻主题、科幻主题的角色和道具)。

七、总结

抽象工厂模式通过将一组相关产品的创建逻辑封装到具体工厂中,提供了一种灵活且可维护的方式来创建对象。它特别适合于需要根据不同环境或配置创建不同产品系列的场景。但在使用时需要权衡其带来的代码复杂度,尤其是在产品结构可能频繁变化的情况下。

相关文章:

Java抽象工厂模式详解

Java 抽象工厂模式是一种创建型设计模式,它提供了一种方式,可以将一组具有共同主题的单个工厂封装起来,而不必指定它们具体的类。这种模式属于创建型模式,它隐藏了对象创建的逻辑,将客户端与具体类的实现解耦。 一、核…...

matlab实现高斯烟羽模型算法

高斯烟羽模型的matlab代码 Code.m , 441 Cross.m , 1329 fit.m , 2080 fitness.m , 2160 fitness1.m , 2191 gaosiyanyu.m , 1936 jixian.m , 169 main.m , 155 mGA.m , 10415 mGA_new.fig , 7218 mGA_new.m , 18196 mPSO.m , 6681 Mutation.m , 1234 point.m , 1976 Select.m…...

SpringBoot parent依赖高版本覆盖低版本问题

问题 在Spring Boot项目中,有时候我们会遇到这样的情况:当我们引入了多个依赖库,而这些库中有相同的依赖项但版本不同。这种情况下,高版本的依赖可能会覆盖低版本的依赖,导致项目运行时出现不期望的行为或错误。为了解…...

OpenCV C/C++ 视频播放器 (支持调速和进度控制)

OpenCV C/C 视频播放器 (支持调速和进度控制) 本文将引导你使用 C 和 OpenCV 库创建一个功能稍复杂的视频播放器。该播放器不仅能播放视频,还允许用户通过滑动条来调整播放速度(加速/减速)以及控制视频的播放进度。 使用opencv打开不会压缩画…...

【Linux庖丁解牛】—自定义shell的编写!

1. 打印命令行提示符 在我们使用系统提供的shell时,每次都会打印出一行字符串,这其实就是命令行提示符,那我们自定义的shell当然也需要这一行字符串。 这一行字符串包含用户名,主机名,当前工作路径,所以&a…...

C++抽象类与多态实战解析

这段 C 代码演示了 抽象类(Abstract Class) 和 多态(Polymorphism) 的使用,它定义了一个表示教师的抽象基类 Teacher,并派生出两个具体的子类:EnglishTeacher(英语老师)和…...

OpenAI API 流式传输

OpenAI API 流式传输教程 🌊 本教程将详细解释 OpenAI API 如何进行数据流式传输,从基本的文本块到复杂的工具调用指令。流式传输允许你逐步从模型接收数据,这对于构建响应灵敏的用户界面和处理长输出非常有用。 1. 基础知识:Ser…...

嵌入式分析利器:DuckDB与SqlSugar实战

​ 一、DuckDB 的核心特性与适用场景 DuckDB 是一款 嵌入式分析型数据库(OLAP) ,专为高效查询设计,主要特点包括: 列式存储与向量化引擎 数据按列存储,提升聚合统计效率(如 SUM/AVG&#xf…...

嵌入式学习笔记 - freeRTOS任务设计要点

一 中断函数中不允许操作任务 因为中断函数使用的上下文环境是MSP环境,而非PSP环境,不允许挂起任务,不允许阻塞任务的任何操作。 可以使用FromISR函数进行操作。 二 中断的频率与处理时间 中断的处理时间要远低于任务的运行时间&#xff…...

Linux运维笔记:1010实验室电脑资源规范使用指南

文章目录 一. 检查资源使用情况,避免冲突1. 检查在线用户2. 检查 CPU 使用情况3. 检查 GPU 使用情况4. 协作建议 二. 备份重要文件和数据三. 定期清理硬盘空间四. 退出 ThinLinc 时注销,释放内存五. 校外使用时配置 VPN注意事项 总结 实验室的电脑配备了…...

12:点云处理—调平,角度,平面度,高度,体积

1.调平 2.夹角、平面度 3.高度、体积...

Marketo 集成 8x8 Connect 短信 API 指南

一、🔍 项目背景与目标 在营销自动化流程中,需要在用户完成特定行为(如填写表单、完成注册)后,自动发送一条短信进行提醒、欢迎或验证。 Marketo 原生不具备短信发送能力,但支持通过 Webhook 集成第三方 A…...

【Docker 从入门到实战全攻略(二):核心概念 + 命令详解 + 部署案例】

5. Docker Compose Docker Compose 是一个用于定义和运行多容器 Docker 应用的工具。通过一个 YAML 文件来配置应用服务,然后使用一个命令即可创建并启动所有服务。 基本命令 docker-compose up # 创建并启动所有服务 docker-compose down # 停止并移除容器、网络等…...

Elasticsearch索引(Index)介绍,它与数据库中的表有什么区别?

在Elasticsearch(ES)中,索引(Index)是存储和组织文档(Document)的逻辑容器,类似于关系型数据库(如MySQL)中的“数据库(Database)”或“表(Table)”,但设计理念和实现机制有显著差异。以下从定义、核心特性、与数据库表的对比三方面详细解析。 一、索引的定义与…...

Elasticsearch中什么是分析器(Analyzer)?它由哪些组件组成?

在Elasticsearch(ES)中,分析器(Analyzer)是处理文本的核心组件,负责将原始文本转换为适合索引和搜索的词项(Term)。它直接影响搜索的准确性和性能,是构建高效搜索系统的关键。 一、分析器的核心作用 1. 分词(Tokenization):将文本拆分为独立的词(Token)。 例如…...

使用 SseEmitter 实现 Spring Boot 后端的流式传输和前端的数据接收

1.普通文本消息的发送和接收 GetMapping("/stream")public SseEmitter streamResponse() {SseEmitter emitter new SseEmitter(0L); // 0L 表示永不超时Executors.newSingleThreadExecutor().execute(() -> {try {for (int i 1; i < 5; i) {emitter.send(&q…...

.net Avalonia 在centos部署

.NET Avalonia 在 CentOS 部署指南 在跨平台应用开发中&#xff0c;.NET Avalonia 凭借其强大的功能和灵活性受到了广泛关注。而将基于 .NET Avalonia 开发的应用程序部署到 CentOS 系统上&#xff0c;是很多开发者会面临的任务。下面就为大家详细介绍在 CentOS 上部署 .NET A…...

MyBatis深度解析:XML/注解配置与动态SQL编写实战

引言 在现代Java企业级应用开发中&#xff0c;MyBatis作为一款优秀的持久层框架&#xff0c;因其灵活性和易用性广受开发者喜爱。相比Hibernate等全自动ORM框架&#xff0c;MyBatis提供了更接近SQL的开发体验&#xff0c;同时又不失面向对象的优雅。本文将深入探讨MyBatis的核…...

面试经验 对常用 LLM 工具链(如 LlamaFactory)的熟悉程度和实践经验

面试场景&#xff1a; 你正在面试一个大型语言模型&#xff08;LLM&#xff09;工程师或研究员的职位&#xff0c;面试官想了解你对常用 LLM 工具链&#xff08;如 LlamaFactory&#xff09;的熟悉程度和实践经验。 面试经验分享&#xff1a;LlamaFactory-CLI 工具实践 面试官…...

【conda配置深度学习环境】

好的&#xff01;我们从头开始配置一个基于Conda的虚拟环境&#xff0c;覆盖深度学习&#xff08;如PyTorch&#xff09;和传统机器学习&#xff08;如XGBoost&#xff09;&#xff0c;并适配你的显卡&#xff08;假设为NVIDIA&#xff0c;若为AMD请告知&#xff09;。以下是完…...

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

文章目录 题目介绍题解 题目介绍 题解 题解链接&#xff1a;题解 核心思路&#xff1a;通过二分查找的确定分割点使左右两部分元素数量相等。 class Solution {public double findMedianSortedArrays(int[] nums1, int[] nums2) {int n1 nums1.length;int n2 nums2.length…...

【相机基础知识与物体检测】更新中

参考&#xff1a; 黑马机器人 | 相机标定&物体检测https://robot.czxy.com/docs/camera/ 01-相机基础 相机基础概述 相机是机器视觉的基础&#xff0c;相机直接产生了相机数据。所有视觉算法都是作用在相机数据上的。相机数据的好坏&#xff0c;或者对相机数据的理解方式…...

【前端】性能优化和分类

本页知识点参考&#xff1a;https://zhuanlan.zhihu.com/p/514222781 1. 加载性能优化 1.1 网站性能优化 content方法&#xff1a; 1&#xff09;减少HTTP请求&#xff1a;合并文件&#xff0c;CSS精灵&#xff0c;inline Image 2&#xff09;减少DNS查询&#xff1a;DNS缓存&…...

PPO和GRPO算法

verl 是现在非常火的 rl 框架&#xff0c;而且已经支持了多个 rl 算法&#xff08;ppo、grpo 等等&#xff09;。 过去对 rl 的理解很粗浅&#xff08;只知道有好多个角色&#xff0c;有的更新权重&#xff0c;有的不更新&#xff09;&#xff0c;也曾硬着头皮看了一些论文和知…...

ceph 对象存储用户限额满导致无法上传文件

查看日志 kl logs -f rook-ceph-rgw-my-store-a-5cc4c4d5b5-26n6j|grep -i error|head -1Defaulted container "rgw" out of: rgw, log-collector, chown-container-data-dir (init) debug 2025-05-30T19:44:11.573+0000 7fa7b7a6d700...

rk3588 上运行smolvlm-realtime-webcam,将视频转为文字描述

smolvlm-realtime-webcam 是一个开源项目&#xff0c;结合了轻量级多模态模型 SmolVLM 和本地推理引擎 llama.cpp&#xff0c;能够在本地实时处理摄像头视频流&#xff0c;生成自然语言描述&#xff0c; 开源项目地址 https://github.com/ngxson/smolvlm-realtime-webcamhttps…...

某航参数逆向及设备指纹分析

文章目录 1. 写在前面2. 接口分析3. 加密分析4. 算法还原5. 设备指纹风控分析与绕过【🏠作者主页】:吴秋霖 【💼作者介绍】:擅长爬虫与JS加密逆向分析!Python领域优质创作者、CSDN博客专家、阿里云博客专家、华为云享专家。一路走来长期坚守并致力于Python与爬虫领域研究…...

SQL思路解析:窗口滑动的应用

目录 &#x1f3af; 问题目标 第一步&#xff1a;从数据中我们能直接得到什么&#xff1f; 第二步&#xff1a;我们想要的“7天窗口”长什么样&#xff1f; 第三步&#xff1a;SQL 怎么表达“某一天的前六天”&#xff1f; &#x1f50d;JOIN 比窗口函数更灵活 第四步&am…...

Rust 学习笔记:Box<T>

Rust 学习笔记&#xff1a;Box Rust 学习笔记&#xff1a;Box<T\>Box\<T> 简介使用 Box\<T\> 在堆上存储数据启用带有 box 的递归类型关于 cons 列表的介绍计算非递归类型的大小使用 Box\<T\> 获取大小已知的递归类型 Rust 学习笔记&#xff1a;Box<…...

C# 从 ConcurrentDictionary 中取出并移除第一个元素

C# 从 ConcurrentDictionary 中取出并移除第一个元素 要从 ConcurrentDictionary<byte, int> 中取出并移除第一个元素&#xff0c;需要结合 遍历 和 原子移除操作。由于 ConcurrentDictionary 是无序集合&#xff0c;"第一个元素" 通常是指最早添加的元素&…...