23种设计模式-创建型模式-抽象工厂

news/2025/4/1 5:17:19/

文章目录

  • 简介
  • 场景
  • 问题
    • 1. 风格一致性失控
    • 2. 对象创建硬编码
    • 3. 产品族管理失效
  • 解决
  • 总结

简介

抽象工厂是一种创建型设计模式,可以生成相关对象系列,而无需指定它们的具体类。

场景

假设你正在写一个家具店模拟器。
你的代码这些类组成:

  1. 相关产品系列,例如:椅子 + 沙发 + 咖啡桌。
  2. 此系列有多种风格。例如,椅子 + 沙发 + 咖啡桌系列有以下风格:现代、维多利亚、装饰艺术。

在这里插入图片描述
你需要一种方法来创建家具对象,确保它们与同一风格的其他对象相匹配。如果客户收到风格不匹配的家具,他们就会非常生气。

在这里插入图片描述
另外,在向程序添加新产品或产品系列时,你也不想更改现有代码。
按照惯例,大家一开始会怎么实现?

java">// 直接实例化具体家具类导致风格混杂
class Client {public void createLivingRoom() {// 混合使用不同风格组件(致命错误)Chair modernChair = new ModernChair();Sofa victorianSofa = new VictorianSofa();CoffeeTable artDecoTable = new ArtDecoCoffeeTable();modernChair.sit();victorianSofa.lieDown();artDecoTable.placeMagazine();}
}// 具体产品实现
class ModernChair extends Chair {public void sit() { System.out.println("Modern chair sitting"); } 
}class VictorianSofa extends Sofa {public void lieDown() { System.out.println("Victorian sofa relaxing"); }
}class ArtDecoCoffeeTable extends CoffeeTable {public void placeMagazine() { System.out.println("ArtDeco table placement"); }
}

问题

1. 风格一致性失控

客户端直接创建不同风格的对象(现代椅子+维多利亚沙发)
╭── 问题场景 ──╮
用户订单要求"维多利亚风格客厅"时:
┌─────────────────┬──────────────────────┐
│ 预期组合 │ 实际可能创建的组合 │
├─────────────────┼──────────────────────┤
│ VictorianChair │ VictorianChair │
│ VictorianSofa │ ModernSofa ←不匹配 │
│ VictorianTable │ ArtDecoTable ←灾难性 │
└─────────────────┴──────────────────────┘
结果:客户收到风格冲突的家具套装

2. 对象创建硬编码

每当新增风格时(如新增ArtDeco),强制修改所有客户端代码

java">
// 新增风格场景产生连锁修改
class Client {// 必须添加新分支判断public void createSet(String style) {if ("ArtDeco".equals(style)) { // 破坏开放封闭原则chair = new ArtDecoChair(); // 需要新增多个类引用sofa = new ArtDecoSofa();}}
}

3. 产品族管理失效

缺乏统一约束机制,易出现类型错误

java">// 错误将现代餐桌与维多利亚椅组合(类型系统无法阻止)
FurnitureSet set = new FurnitureSet(new ModernChair(),new VictorianDiningTable() // 应该抛出异常但现有代码无法约束
);

解决

抽象工厂模式建议的第一件事就是明确声明产品系列中每个不同产品的接口(例如,椅子、沙发或咖啡桌)。然后,让所有风格的产品都实现这些接口。例如,所有风格的椅子都可以实现 Chair 接口;所有风格的咖啡桌都可以实现 CoffeeTable 接口,等等。

在这里插入图片描述

java">// 接口约束产品规格
public interface Chair {void sit();
}public interface Sofa {void lieDown();
}public interface CoffeeTable {void placeItem();
}
// 确保现代系列组件统一
public class ModernChair implements Chair {@Overridepublic void sit() { System.out.println("Modern chair designed seating"); }
}public class ModernSofa implements Sofa {@Overridepublic void lieDown() { System.out.println("Modern sofa clean lines design"); }
}public class ModernCoffeeTable implements CoffeeTable {@Overridepublic void placeItem() { System.out.println("Modern geometric table surfaces"); }
}// 保证维多利亚风格一致性
public class VictorianChair implements Chair {@Overridepublic void sit() { System.out.println("Classic carved wood chair"); }
}public class VictorianSofa implements Sofa {@Overridepublic void lieDown() { System.out.println("Antique fabric sofa"); }
}public class VictorianCoffeeTable implements CoffeeTable {@Overridepublic void placeItem() { System.out.println("Ornate marble-top table"); }
}

下一步是声明抽象工厂(接口),其中包含特定产品系列所有产品的创建方法列表(例如,createChair、createSofa 和 createCoffeeTable)。这些方法必须返回我们之前定义的抽象产品类型接口:Chair、Sofa、CoffeeTable 等等。对于产品的每种风格,我们基于 AbstractFactory 接口创建一个单独的工厂类。这个工厂类是返回特定类型产品的类。例如,ModernFurnitureFactory 只能创建 ModernChair、ModernSofa 和 ModernCoffeeTable 对象。
在这里插入图片描述

java">public interface FurnitureFactory {Chair createChair();Sofa createSofa();CoffeeTable createCoffeeTable();
}// 现代风格产品线工厂
public class ModernFactory implements FurnitureFactory {@Overridepublic Chair createChair() { return new ModernChair(); }@Overridepublic Sofa createSofa() { return new ModernSofa(); }@Overridepublic CoffeeTable createCoffeeTable() { return new ModernCoffeeTable(); }
}// 维多利亚风格产品线工厂
public class VictorianFactory implements FurnitureFactory {@Overridepublic Chair createChair() { return new VictorianChair(); }@Overridepublic Sofa createSofa() { return new VictorianSofa(); }@Overridepublic CoffeeTable createCoffeeTable() { return new VictorianCoffeeTable(); }
}

客户端代码必须通过抽象接口来和工厂、产品协作。这样就可以更改传给客户端代码的工厂的类型以及客户端代码接收的产品风格,而不破坏实际的客户端代码。
假设客户希望工厂生产一把椅子。客户不必知道工厂的类别,也不必关心它得到的椅子是什么类型。无论是现代风格还是维多利亚风格的椅子,客户都必须使用抽象Chair接口以相同的方式处理所有椅子。唯一知道的是sitOn方法。此外,无论返回哪种椅子,它总是与同一工厂对象生产的沙发或咖啡桌风格相匹配。

java">public class FurnitureStore {private FurnitureFactory factory;// 动态绑定具体工厂public FurnitureStore(FurnitureFactory factory) {this.factory = factory;}public void showcaseSet() {Chair chair = factory.createChair();Sofa sofa = factory.createSofa();CoffeeTable table = factory.createCoffeeTable();System.out.println("展示完整风格套件:");chair.sit();sofa.lieDown();table.placeItem();}
}// 使用示例
public class Main {public static void main(String[] args) {// 创建现代风格商店FurnitureStore modernStore = new FurnitureStore(new ModernFactory());modernStore.showcaseSet();// 创建维多利亚风格商店FurnitureStore victorianStore = new FurnitureStore(new VictorianFactory());victorianStore.showcaseSet();}
}

还有一件事需要明确:如果客户端只接触抽象接口,那是什么创建了实际的工厂对象(即new ModernFactory())?通常,应用程序在初始化阶段创建一个具体的工厂对象。在这之前,应用程序必须根据配置或环境设置选择工厂类型。

总结

在这里插入图片描述

  1. 抽象产品(Abstract Prod­uct):构成所有产品的一组接口。
  2. 具体产品(Con­crete Prod­uct):抽象产品的多种不同类型实现。所有风格产品(维多利亚/现代)都必须实现相应的抽象产品(椅子/沙发)。
  3. 抽象工厂(Abstract Fac­to­ry)接口:声明了一组创建各种抽象产品的方法。
  4. 具体工厂(Con­crete Fac­to­ry):实现抽象工厂的产品创建方法。每个具体工厂都对应特定风格的产品,且只能创建这一种风格的产品。
  5. 尽管具体工厂会对具体产品进行初始化,它的创建方法必须返回相应的抽象产品。只有这样,使用工厂类的客户端代码就不会与工厂创建的特定风格产品耦合。客户端(Client)只需通过抽象接口调用工厂和产品对象,就可以跟任何具体工厂/产品进行交互。

http://www.ppmy.cn/news/1582739.html

相关文章

C#与西门子PLC的六大通信库

C#与西门子PLC的六大通信库: 一、S7.NET S7.NET是一款开源的S7协议通信库,支持西门子S7通信。 二、Sharp7 Sharp7与S7.NET一样,是一款.NET版本的S7通信库。 三、Snap7 Snap7是一个开源的C通信库,支持西门子S7通信。 四、Prodave P…

31天Python入门——第9天:再学函数

你好,我是安然无虞。 文章目录 再学函数1. 变量在函数中的作用域2. 函数的参数传递.补充学习: 不定长参数*args和**kwargs 3. 值传递和引用传递补充学习: 把函数作为参数传递 4. 匿名函数5. python中内置的常用函数zip()map()filter()all()any() 6. 函数练习 再学函…

DeepSeek面试——模型架构和主要创新点

本文将介绍DeepSeek的模型架构多头潜在注意力(MLA)技术,混合专家(MoE)架构, 无辅助损失负载均衡技术,多Token 预测(MTP)策略。 一、模型架构 DeepSeek-R1的基本架构沿用…

【CVPR2024-工业异常检测】PromptAD方法(CLIP和提示学习)

Preliminaries 3.1. CLIP and Prompt Learning(CLIP和提示学习) CLIP核心机制 输入:未知图像 多组文本提示(如 “a photo of [class]”) 操作: 图像编码:视觉编码器 f() 提取图像特征 f(i)文本编码:文…

IDEA批量替换项目下所有文件中的特定内容

文章目录 1. 问题引入2. 批量替换项目下所有文件中的特定内容2.1 右键项目的根目录,点击在文件中替换2.2 输入要替换的内容 3. 解决替换一整行文本后出现空行的问题4. 增加筛选条件提高匹配的精确度 更多 IDEA 的使用技巧可以查看 IDEA 专栏: IDEA 1. 问…

补充--HTTP常见的状态码

1xx(信息性状态码) - 表示接收的进程正在请求中,客户端应继续其操作。 100 Continue(继续): 客户端应该继续其请求。 101 Switching Protocols(切换协议): 服务器根据客户端的请求切换协议。 …

快速查询手机是否处于联网状态?

手机是否处于联网状态对于我们日常生活中的沟通、工作和娱乐都至关重要。有时候我们需要迅速了解一个手机号码的在网状态,例如是正常使用、停机、不在网等。而要实现这一功能,我们可以利用挖数据平台提供的在线查询工具,通过API接口来查询手机…

macos设置docker可以ping通容器

今天在启动项目时候seata报错: io.seata.common.exception.FrameworkException: can not connect to services-server 意思就是连接不上seata server,然后打开注册中心一看 ip为:172.17.0.5,果断ping一下 原因 ping不通的原因是&#xff0…