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

「软件设计模式」工厂方法模式 vs 抽象工厂模式

前言

在软件工程领域,设计模式是解决常见问题的经典方案。本文将深入探讨两种创建型模式:工厂方法模式抽象工厂模式,通过理论解析与实战代码示例,帮助开发者掌握这两种模式的精髓。


一、工厂方法模式(Factory Method Pattern)

1.1 模式思想

工厂方法模式的核心在于将对象的创建延迟到子类,通过定义创建对象的接口,让子类决定实例化哪个类。这种模式完美体现了依赖倒置原则

1.2 模式结构

  • Product:抽象产品接口
  • ConcreteProduct:具体产品实现
  • Creator:抽象创建者
  • ConcreteCreator:具体创建者

1.3 代码示例

#include <iostream>
#include <string>class Shape {
public:virtual void draw() = 0;
};class Circle : public Shape {void draw() {std::cout << "Circle::draw()" << std::endl;}
};class Rectangle : public Shape {void draw() {std::cout << "Rectangle::draw()" << std::endl;}
};class Square : public Shape {void draw() {std::cout << "Square::draw()" << std::endl;}
};class ShapeFactory {
public:Shape* getShape(const std::string& shapeType) {if (shapeType.empty()) {return nullptr;}if (shapeType == "CIRCLE") {return new Circle();} else if (shapeType == "RECTANGLE") {return new Rectangle();} else if (shapeType == "SQUARE") {return new Square();}return nullptr;}
};
#include "factory_mode.h"int main(int argc, char const* argv[]) {ShapeFactory* shapeFactory = new ShapeFactory();Shape* shape1 = shapeFactory->getShape("CIRCLE");shape1->draw();Shape* shape2 = shapeFactory->getShape("RECTANGLE");shape2->draw();Shape* shape3 = shapeFactory->getShape("SQUARE");shape3->draw();delete shapeFactory;return 0;
}

1.4 运行结果

1.5 适用场景

  • 需要灵活扩展产品类型
  • 创建过程需要封装处理逻辑
  • 需要解耦客户端与具体产品类

二、抽象工厂模式(Abstract Factory Pattern)

2.1 模式思想

抽象工厂模式通过创建相关对象族来提升系统的扩展性,强调产品系列的概念。它比工厂方法模式更高层次的抽象。

2.2 模式结构

  • AbstractFactory:抽象工厂接口
  • ConcreteFactory:具体工厂实现
  • AbstractProduct:抽象产品接口
  • ConcreteProduct:具体产品实现

2.3 代码示例

产品定义:

#include <iostream>
#include <string>
// 
class Color {
public:virtual void fill() = 0;
};class Red : public Color {
public:void fill() {std::cout << "Red::fill()" << std::endl;}
};class Green : public Color {
public:void fill() {std::cout << "Green::fill()" << std::endl;}
};class Blue : public Color {
public:void fill() {std::cout << "Blue::fill()" << std::endl;}
};
// 形状产品
class Shape {
public:virtual void draw() = 0;
};class Circle : public Shape {void draw() {std::cout << "Circle::draw()" << std::endl;}
};class Rectangle : public Shape {void draw() {std::cout << "Rectangle::draw()" << std::endl;}
};class Square : public Shape {void draw() {std::cout << "Square::draw()" << std::endl;}
};

 抽象工厂

#include "color.h"
#include "shape.h"
// 抽象工厂
class AbstractFactory {
public:virtual Shape* getShape(const std::string& shapeType) = 0;virtual Color* getColor(const std::string& colorType) = 0;
};

 实体工厂

#include "abstract_factory.h"
// 实体工厂类// 形状工厂
class ShapeFactory : public AbstractFactory {
public:Shape* getShape(const std::string& shapeType) {if (shapeType.empty()) {return nullptr;}if (shapeType == "CIRCLE") {return new Circle();} else if (shapeType == "RECTANGLE") {return new Rectangle();} else if (shapeType == "SQUARE") {return new Square();}return nullptr;}Color* getColor(const std::string& colorType) {return nullptr;}
};// 颜色工厂
class ColorFactory : public AbstractFactory {
public:Color* getColor(const std::string& colorType) {if (colorType.empty()) {return nullptr;}if (colorType == "RED") {return new Red();} else if (colorType == "GREEN") {return new Green();} else if (colorType == "BLUE") {return new Blue();}return nullptr;}Shape* getShape(const std::string& shapeType) {return nullptr;}
};

 生产商:

#include "factory.h"class FactoryProducer {
public:static AbstractFactory* getFactory(const std::string& choice) {if (choice == "SHAPE") {return new ShapeFactory();} else if (choice == "COLOR") {return new ColorFactory();}return nullptr;}
};

main:

#include "producer.h"// 抽象工厂 demo
int main(int argc, char const* argv[]) {AbstractFactory* shapeFactory = FactoryProducer::getFactory("SHAPE");Shape* shape1 = shapeFactory->getShape("CIRCLE");shape1->draw();Shape* shape2 = shapeFactory->getShape("RECTANGLE");shape2->draw();Shape* shape3 = shapeFactory->getShape("SQUARE");shape3->draw();AbstractFactory* colorFactory = FactoryProducer::getFactory("COLOR");Color* color1 = colorFactory->getColor("RED");color1->fill();Color* color2 = colorFactory->getColor("GREEN");color2->fill();Color* color3 = colorFactory->getColor("BLUE");color3->fill();delete shapeFactory;delete colorFactory;return 0;
}

2.4 运行结果

2.5 适用场景

  • 需要创建多个相关对象组成的系列
  • 系统需要支持不同产品族的切换
  • 产品对象之间存在约束关系

三、核心差异对比

维度工厂方法模式抽象工厂模式
抽象层次单个产品创建产品族创建
扩展方向垂直扩展(新增产品类型)水平扩展(新增产品族)
实现方式继承组合
系统复杂度简单复杂
典型应用场景日志记录器、支付方式跨平台UI组件、数据库访问

四、模式选择指南

  • 选择工厂方法模式当:

    • 需要解耦客户端与具体产品类
    • 系统需要支持多种同类产品的创建
    • 产品类型相对单一
  • 选择抽象工厂模式当:

    • 需要创建多个相互关联的产品
    • 需要保证产品之间的兼容性
    • 系统需要支持不同产品族的切换

五、最佳实践建议

  1. 优先使用工厂方法:当产品结构简单时,避免过度设计
  2. 注意开闭原则:通过扩展而非修改来增加新产品
  3. 使用依赖注入:结合Spring等框架实现更灵活的工厂管理
  4. 文档化产品族:明确各产品之间的约束关系
  5. 性能考量:复杂工厂实现需要考虑对象池等优化手段

结语

        掌握工厂模式是成为架构师的重要阶梯。工厂方法模式像专业工匠,专注单一产品的精雕细琢;抽象工厂模式如生产总监,统筹协调整个产品家族。理解它们的差异,才能在系统设计中做出最合适的选择。 


http://www.mrgr.cn/news/90696.html

相关文章:

  • clickhouse replicatedmergetree 恢复
  • 《8天入门Trustzone/TEE/安全架构》
  • Mac 下使用多版本 Node
  • 并发编程 - 线程同步(七)之互斥锁Monitor
  • Modbus TCP协议
  • Linux第106步_Linux内核RTC驱动实验
  • 【Unity Shader编程】之顶点着色器
  • 「软件设计模式」单例模式
  • linux的三剑客和进程处理
  • 200个Python练手项目源码
  • rocketmq-netty通信设计-request和response
  • 【DeepSeek】Deepseek辅组编程-通过卫星轨道计算终端距离、相对速度和多普勒频移
  • 【Stable Diffusion部署至GNU/Linux】安装流程
  • Ubuntu启动geteck/jetlinks实战:Docker启动
  • Proxmox VE 8.3 qm 方式导入ESXi Linux OVA UEFI模式虚拟机
  • C++类和对象进阶:运算符重载深度详解
  • 在vscode中拉取gitee里的项目并运行
  • Python----PyQt开发(PyQt高级:手搓一个文件浏览器)
  • Druid GetConnectionTimeoutException解决方案之一
  • vue-model如何自定义指令,及批量注册自定义指令
  • 【GRPO】GRPO原理原文翻译
  • docker学习---第3步:docker实操大模型
  • Hive增量迁移方案与实操PB级
  • Linux初始化 配置yum源
  • 大数据学习之PB级百战出行网约车二
  • poi 将图片写入到excel文件中