Java二十三种设计模式-抽象工厂模式(3/23)

抽象工厂模式:复杂系统的灵活构建者

引言

在软件开发中,抽象工厂模式是一种提供接口以创建相关或依赖对象族的创建型设计模式。这种模式允许客户端使用一个共同的接口来创建不同的产品族,而无需指定具体类。

基础知识,java设计模式总体来说设计模式分为三大类:

(1)创建型模式,共5种:工厂方法模式、抽象工厂模式、单例模式、建造者模式、原型模式。

(2)结构型模式,共7种:适配器模式、装饰器模式、代理模式、外观模式、桥接模式、组合模式、享元模式。

(3)行为型模式,共11种:策略模式、模板方法模式、观察者模式、迭代子模式、责任链模式、命令模式、备忘录模式、状态模式、访问者模式、中介者模式、解释器模式。

第一部分:抽象工厂模式概述

1.1 定义与目的

抽象工厂模式的基本定义

抽象工厂模式是一种创建型设计模式,用于创建一系列相关或相互依赖的对象,而不需要指定它们具体的类。这种模式提供了一个接口,用于生成一组相关的对象,而客户端不需要知道这些对象的具体类。

解释为何需要抽象工厂模式

在复杂的系统中,对象的创建可能会涉及到多个相关的类,这些类之间可能存在一定的依赖关系。如果直接在客户端代码中创建这些对象,会导致客户端与具体类紧密耦合,难以扩展和维护。抽象工厂模式通过提供一个抽象的创建接口,将对象的创建过程封装起来,使得系统更加灵活和易于扩展。

1.2 组成元素

抽象工厂(Abstract Factory)

  • 定义了创建一系列相关或依赖对象的接口。
  • 它是一个抽象角色,不实现具体的创建逻辑。

具体工厂(Concrete Factory)

  • 实现了抽象工厂的接口,生成具体的产品对象。
  • 每个具体工厂都对应一个产品族。

抽象产品(Abstract Product)

  • 定义了产品的接口,是所有具体产品类的共同父类。
  • 它是一个抽象角色,不实现具体的产品类。

具体产品(Concrete Product)

  • 实现了抽象产品的接口,是抽象工厂模式中被创建的具体对象。
  • 每个具体产品都属于一个产品族。

客户端(Client)

  • 使用抽象工厂来请求创建对象,与具体工厂和具体产品解耦。
  • 客户端通过抽象工厂的接口与具体工厂交互,从而获得所需的产品。

角色之间的关系

  • 抽象工厂与具体工厂:具体工厂实现了抽象工厂的接口,负责创建具体的产品。
  • 抽象产品与具体产品:具体产品实现了抽象产品的接口,是系统中实际使用的对象。
  • 客户端与抽象工厂:客户端通过抽象工厂的接口请求产品,而不直接与具体工厂或具体产品交互。

抽象工厂模式的核心优势在于其封装性、灵活性和可扩展性。通过使用抽象工厂模式,可以在不修改现有代码的基础上,引入新的产品族,满足开闭原则。在下一部分中,我们将通过Java代码示例来展示抽象工厂模式的具体实现。

 

第二部分:抽象工厂模式实现

2.1 Java实现示例

以下是使用Java语言实现抽象工厂模式的一个示例。假设我们有一个形状和颜色的工厂,它们可以生成多种类型的形状和颜色。

// 抽象产品:形状
interface Shape {void draw();
}// 具体产品:圆形
class Circle implements Shape {public void draw() {System.out.println("Drawing a Circle");}
}// 抽象产品:颜色
interface Color {void fill();
}// 具体产品:红色
class Red implements Color {public void fill() {System.out.println("Filling with Red Color");}
}// 抽象工厂
interface Factory {Shape getShape();Color getColor();
}// 具体工厂:形状和颜色工厂
class ShapeColorFactory implements Factory {private String colorType;public ShapeColorFactory(String colorType) {this.colorType = colorType;}@Overridepublic Shape getShape() {return new Circle(); // 假设这个工厂只能生成圆形}@Overridepublic Color getColor() {if ("Red".equalsIgnoreCase(colorType)) {return new Red();}return new Color() { // 默认颜色public void fill() {System.out.println("Filling with Default Color");}};}
}// 客户端代码
public class Client {public static void main(String[] args) {Factory factory = new ShapeColorFactory("Red");Shape shape = factory.getShape();shape.draw();Color color = factory.getColor();color.fill();}
}

2.2 设计原则与模式应用

抽象工厂模式如何体现设计原则

开闭原则

抽象工厂模式遵循开闭原则,即软件实体应对扩展开放,对修改封闭。当需要添加新的形状或颜色时,我们只需添加相应的具体产品类和具体工厂类,而无需修改现有的抽象工厂和客户端代码。

里氏替换原则

在抽象工厂模式中,具体工厂实现了抽象工厂的接口,确保了具体产品能够替换其抽象产品。这意味着客户端可以接收任何从抽象工厂派生的实例,并期待它仍然有效。

依赖倒置原则

抽象工厂模式通过依赖于抽象(接口或抽象类)而不是具体实现,从而减少了客户端与具体类的耦合。客户端与抽象工厂交互,而不是与具体工厂或具体产品直接交互。

接口隔离原则

虽然抽象工厂模式提供了一个创建相关对象的接口,但在设计时应谨慎考虑接口的职责,避免将不相关的创建方法放在同一个接口中,以符合接口隔离原则。

抽象工厂模式在设计时需要仔细考虑产品族的划分和工厂接口的设计,以确保系统的灵活性和可扩展性。在下一部分中,我们将探讨抽象工厂模式的使用场景。

第三部分:抽象工厂模式使用场景

3.1 产品族的多样化

何时产品族具有多样化特征

产品族指的是一组具有共同主题或共享通用接口的产品。当这些产品需要根据不同的场景或条件以不同的方式进行组合时,产品族就呈现出多样化特征。

  • 例子:考虑一个图形界面库,它可能包含多种形状(如圆形、矩形、三角形)和多种颜色(如红色、蓝色、绿色)。不同的应用程序可能需要不同的形状和颜色组合,例如,一个儿童绘画程序可能需要鲜艳的颜色和简单的形状。

适合使用抽象工厂模式的情况

  • 当存在多个产品族,并且每个产品族中的产品需要一起使用时。
  • 当需要提供产品族的不同变体,以适应不同的环境或条件时。

抽象工厂模式的应用

  • 通过定义一个抽象工厂接口,不同的具体工厂可以实现这个接口,以创建特定产品族中的对象。
  • 客户端代码通过抽象工厂接口与具体工厂交互,请求所需的产品族,而无需关心具体的产品实现。

3.2 系统之间的依赖关系

系统间依赖关系的问题

在大型软件系统中,不同的组件或模块之间可能存在依赖关系。如果这些依赖关系处理不当,可能会导致代码难以维护和扩展。

抽象工厂模式如何进行有效管理

  • 解耦:抽象工厂模式通过将对象创建的逻辑集中到具体的工厂类中,降低了模块间的直接依赖。
  • 配置化:系统可以通过配置或参数化的方式,动态选择使用哪个具体工厂,从而适应不同的运行时条件。

应用实例

  • 主题定制:在一个支持主题定制的应用程序中,可以为不同的主题定义不同的工厂,每个工厂负责创建符合该主题的视觉元素。
  • 数据库连接:在需要连接多种数据库系统的应用程序中,可以为每种数据库类型定义一个具体的工厂,而客户端则通过抽象工厂接口请求数据库连接。

通过抽象工厂模式,可以在不同的系统或组件之间建立一个清晰且灵活的接口,使得它们能够更容易地协同工作,同时保持各自的独立性和可扩展性。在下一部分中,我们将讨论抽象工厂模式的优点与缺点。

第三部分:抽象工厂模式使用场景

3.1 产品族的多样化

在软件开发中,产品族指的是一组具有共同接口但实现不同的类。当这些类需要根据不同的情境或配置以不同的方式进行组合时,就表现为产品族的多样化。

何时产品族具有多样化特征:

  • 当存在多个产品系列,每个系列都有多种产品变体时。
  • 当产品之间存在依赖关系,需要一起使用以保证一致性时。

适合使用抽象工厂模式的情况:

  • 当需要创建的产品族具有多种变体,且这些变体在结构上相关联时。
  • 当系统需要支持多种主题或外观,例如不同的GUI主题。

3.2 系统之间的依赖关系

系统间的依赖关系管理是软件设计中的一个关键问题。抽象工厂模式通过将创建逻辑封装在工厂类中,有助于降低系统间的耦合度。

系统间依赖关系的问题:

  • 当一个系统组件直接依赖于另一个组件的具体实现时,会导致组件之间的耦合度过高。

抽象工厂模式如何进行有效管理:

  • 通过定义一个抽象工厂接口,不同的系统可以提供各自的具体工厂实现,从而生产出符合各自需求的产品。
  • 客户端代码通过抽象工厂接口与具体工厂解耦,可以灵活地替换具体的工厂实现,以适应不同的业务需求。

第四部分:抽象工厂模式的优点与缺点

4.1 优点

灵活性和可扩展性:

  • 抽象工厂模式允许系统在不修改现有代码的基础上引入新的产品族,提高了系统的可扩展性。

解耦:

  • 客户端代码与具体工厂实现解耦,降低了模块间的依赖。

一致性:

  • 确保了产品族中的对象是一起创建的,从而保证了对象之间的一致性。

4.2 缺点

系统复杂度增加:

  • 每增加一个新的产品族,都需要增加一个新的具体工厂类和产品类,可能会导致系统中类的数量急剧增加。

难以维护:

  • 随着产品族的增加,抽象工厂模式可能会导致系统结构变得更加复杂,难以理解和维护。

不够灵活:

  • 如果系统需要支持多种产品族的任意组合,抽象工厂模式可能不如其他模式(如建造者模式)灵活。

在实际应用中,选择抽象工厂模式需要权衡其优点和缺点。如果系统中存在多个产品族,并且这些产品族需要一起使用,抽象工厂模式是一个不错的选择。然而,如果系统需要高度的灵活性和扩展性,可能需要考虑其他设计模式。在下一部分中,我们将比较抽象工厂模式与其他设计模式,并提供一些最佳实践和建议。

第五部分:抽象工厂模式与其他模式的比较

5.1 与工厂方法模式的比较

工厂方法模式

  • 定义:定义一个用于创建对象的接口,让子类决定实例化哪一个类。
  • 使用场景:当需要创建的对象类型相对较少,且这些对象的创建逻辑可以封装在一个共同的工厂类中。

抽象工厂模式

  • 定义:提供一个创建一系列相关或相互依赖对象的接口,而不需要具体指定它们的类。
  • 使用场景:当存在多个产品族,且这些产品族需要一起使用时。

不同点

  • 产品种类:工厂方法模式通常用于创建单一对象,而抽象工厂模式用于创建相关对象的族。
  • 扩展性:抽象工厂模式在添加新的产品族时更加灵活,而工厂方法模式更适合对象创建逻辑相对固定的情况。

5.2 与建造者模式的对比

建造者模式

  • 定义:用于创建一个复杂的对象,同时允许用户只通过指定复杂对象的类型和内容就能构建它们。
  • 使用场景:当对象的创建过程涉及多个步骤,或者对象的创建逻辑较为复杂时。

抽象工厂模式

  • 定义:用于创建一系列相关或相互依赖的对象。
  • 使用场景:当需要同时创建多个相关对象,并且这些对象的创建逻辑可以封装在一个共同的工厂类中。

不同点

  • 复杂性:建造者模式适用于构建过程复杂或步骤繁多的对象,而抽象工厂模式适用于创建对象族。
  • 灵活性:建造者模式在构建过程中提供了更多的控制和灵活性,而抽象工厂模式则强调了对象族的一致性。

第六部分:最佳实践和建议

6.1 使用抽象工厂模式的最佳时机

  • 产品族多样化:当系统中存在多个产品族,且这些产品族需要根据不同的配置或条件进行变化时。
  • 系统解耦:当需要降低系统组件之间的耦合度,提高模块的独立性和可替换性时。

6.2 避免滥用抽象工厂模式

  • 过度扩展:避免在产品族非常简单或不相关的情况下使用抽象工厂模式,这可能会导致不必要的系统复杂性。
  • 难以维护:随着产品族的增加,管理大量的工厂类和产品类可能会变得困难。

6.3 替代方案

原型模式

  • 定义:通过复制现有的对象来创建新的实例。
  • 适用场景:当创建新对象的成本较高,或者需要快速复制现有对象时。

依赖注入

  • 定义:通过外部提供依赖对象,而不是在类内部创建。
  • 好处:提高了代码的可测试性和灵活性。

服务定位器模式

  • 定义:用于查找和访问服务的机制。
  • 适用场景:当系统中有大量的服务需要被访问,并且服务实例的创建和管理较为复杂时。

抽象工厂模式是一种强大的设计模式,适用于创建相关或依赖对象族的场景。然而,合理选择使用时机和避免滥用同样重要。了解替代方案可以帮助开发者根据具体需求选择最合适的设计模式。

结语

抽象工厂模式是处理复杂系统中对象创建问题的有效工具。通过本文的深入分析,希望读者能够对抽象工厂模式有更全面的理解,并在实际开发中做出合理的设计选择。

 

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/web/46248.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

算法 —— LRU算法

算法 —— LRU算法 LRULRU算法的工作原理:实现方法:性能考虑: 模拟过程splice函数对于std::list和std::forward_list基本语法:功能描述: 示例:注意事项: 如果大家已经学习过了Cache的替换算法和…

ElementUIV12相关使用方法

今日内容 零、 复习昨日 零、 复习昨日 一、Element UI Element,一套为开发者、设计师和产品经理准备的基于 Vue 2.0 的桌面端组件库 官网: https://element.eleme.cn/#/zh-CN Element Plus,基于 Vue 3,面向设计师和开发者的组件库 官网: htt…

C语言--递归

曾经有一个段子:上大学时,我们的c语言老师说:学c时,如果有50%的同学死在了循环上面,那么就有90%的同学死在了递归上面。接下来,就来看看递归是怎么个事? 一.递归的介绍 递归是指一个函数直接或…

跨平台WPF音乐商店应用程序

目录 一 简介 二 设计思路 三 源码 一 简介 支持在线检索音乐,支持实时浏览当前收藏的音乐及音乐数据的持久化。 二 设计思路 采用MVVM架构,前后端分离,子界面弹出始终位于主界面的中心。 三 源码 视窗引导启动源码: namesp…

MySQL(8)事务

目录 1.事务; 1.事务: 1.1 如果CURD不加限制会这么样子? 可能造成数据同时被修改, 数据修改的结果是未知的.(可以想一下之前的抢票线程问题) 1.2 事务概念: 事务就是一组DML语句组成,这些语句在逻辑上存在相关性,这一组DML语句要么全部成功&#xff0…

基于python旅游景点满意度分析设计与实现

1.1研究背景与意义 1.1.1研究背景 随着旅游业的快速发展,满意度分析成为评估旅游景点质量和提升游客体验的重要手段。海口市作为中国的旅游城市之一,其旅游景点吸引了大量游客。然而,如何科学评估和提升海口市旅游景点的满意度,…

【概率论三】参数估计:点估计(矩估计、极大似然法)、区间估计

文章目录 一. 点估计1. 矩估计法2. 极大似然法2.1. 似然函数2.2. 极大似然估计法 3. 评价估计量的标准3.1. 无偏性3.2. 有效性3.3. 一致性 二. 区间估计1. 区间估计的概念2. 正态总体参数的区间估计 参数估计讲什么 由样本来确定未知参数参数估计分为点估计与区间估计 一. 点估…

算法:二叉树相关

目录 题目一:单值二叉树 题目二:二叉树的最大深度 题目三:相同的树 题目四:对称二叉树 题目五:另一棵树的子树 题目六:二叉树的前序遍历 题目七:二叉树遍历 题目八:根据二叉…

linux搭建mysql主从复制(一主一从)

目录 0、环境部署 1、主服务器配置 1.1 修改mysql配置文件 1.2 重启mysql 1.3 为从服务器授权 1.4 查看二进制日志坐标 2、从服务器配置 2.1 修改mysql配置文件 2.2 重启mysql 2.3 配置主从同步 2.4 开启主从复制 3、验证主从复制 3.1 主服务器上创建test…

4款良心软件,免费又实用,内存满了都舍不得卸载

以下几款高质量软件,若是不曾体验,实在是遗憾可惜。 PDF Guru 这是一款开源免费的PDF编辑软件,打开之后功能一目了然。 可以拆分、合并PDF,也可以给PDF添加水印和密码,同时也可以去除别人PDF里的水印或密码&#xff0…

HouseCrafter:平面草稿至3D室内场景的革新之旅

在室内设计、房地产展示和影视布景设计等领域,将平面草稿图快速转换为立体的3D场景一直是一个迫切的需求。HouseCrafter,一个创新的AI室内设计方案,正致力于解决这一挑战。本文将探索HouseCrafter如何将这一过程自动化并提升至新的高度。 一、定位:AI室内设计的革新者 Ho…

【iOS】类对象的结构分析

目录 对象的分类object_getClass和class方法isa流程和继承链分析isa流程实例验证类的继承链实例验证 类的结构cache_t结构bits分析实例验证属性properties方法methods协议protocolsro类方法 类结构流程图解 对象的分类 OC中的对象主要可以分为3种:实例对象&#xf…

TDesign组件库日常应用的一些注意事项

【前言】Element(饿了么开源组件库)在国内使用的普及率和覆盖率高于TDesign-vue(腾讯开源组件库),这也导致日常开发遇到组件使用上的疑惑时,网上几乎搜索不到其文章解决方案,只能深挖官方文档或…

C++右值引用和移动语义

目录 概念: 左值引用和右值引用 概念: 注意: 左值引用的意义 作函数参数 函数引用返回 右值引用的意义 诞生背景 移动构造 移动赋值 其他应用 万能引用和完美转发 默认的移动构造和移动赋值 概念: 左值:顾…

VulnHub:CK00

靶场搭建 靶机下载地址:CK: 00 ~ VulnHub 下载后,在vmware中打开靶机。 修改网络配置为NAT 处理器修改为2 启动靶机 靶机ip扫描不到的解决办法 靶机开机时一直按shift或者esc直到进入GRUB界面。 按e进入编辑模式,找到ro,修…

思路|如何利用oneNote钓鱼?

本文仅用于技术研究学习,请遵守相关法律,禁止使用本文所提及的相关技术开展非法攻击行为,由于传播、利用本文所提供的信息而造成任何不良后果及损失,与本账号及作者无关。 本文来源无问社区,更多实战内容,…

[python]pycharm设置清华源

国内镜像源有以下几个,因为都是国内的,基本速度差不了太多。 清华:https://pypi.tuna.tsinghua.edu.cn/simple 阿里云:http://mirrors.aliyun.com/pypi/simple/ 中国科技大学 https://pypi.mirrors.ustc.edu.cn/simple/ 豆瓣&…

爬虫案例(读书网)(下)

上篇链接: CSDN-读书网https://mp.csdn.net/mp_blog/creation/editor/139306808 可以看见基本的全部信息:如(author、bookname、link.....) 写下代码如下: import requests from bs4 import BeautifulSoup from lxml import etreeheaders{…

scottplot5 中 使用signalXY图,如何更新数据?

🏆本文收录于《CSDN问答解答》专栏,主要记录项目实战过程中的Bug之前因后果及提供真实有效的解决方案,希望能够助你一臂之力,帮你早日登顶实现财富自由🚀;同时,欢迎大家关注&&收藏&…

Apache POI 使用Java处理Excel数据 进阶

1.POI入门教程链接 http://t.csdnimg.cn/Axn4Phttp://t.csdnimg.cn/Axn4P建议&#xff1a;从入门看起会更好理解POI对Excel数据的使用和处理 记得引入依赖&#xff1a; <dependency><groupId>org.apache.poi</groupId><artifactId>poi</artifactI…