23种设计模式 - 抽象工厂模式(Abstract Factory)

抽象工厂模式(Abstract Factory)------ 整套买,不散卖

大白话解释

你去宜家买家具,有两种风格:

  • 北欧风套装
  • 现代风套装

每一套都包含:

  • 沙发(Sofa)
  • 茶几(CoffeeTable)

👉 你只需要选风格 ,工厂自动给你整套家具,绝不混搭

抽象工厂模式:提供一个接口,用来创建一系列相关或相互依赖的对象,而不需要指定具体类。简单说就是------工厂的工厂,生产的是"一家子"产品。

常见场景:

  • 跨平台 UI(Windows 主题 vs Mac 主题,按钮/滚动条/文本框风格要统一)
  • 数据库访问层(MySQL 套件 vs PostgreSQL 套件)
  • 游戏皮肤系统(沙漠主题 vs 雪地主题,所有元素风格统一)

和工厂方法的区别

工厂方法 抽象工厂
生产一种产品 生产一族产品(多种)
一个工厂方法 多个工厂方法
"给我造个日志器" "给我造一整套 UI 组件"

C++ 代码示例

场景:跨平台 UI 组件库,支持 Windows 风格和 Mac 风格。

cpp 复制代码
#include <iostream>
#include <memory>
#include <string>

// ================================
// 抽象产品族:按钮、复选框
// ================================
class Button {
public:
    virtual ~Button() = default;
    virtual void render() = 0;
    virtual void onClick() = 0;
};

class Checkbox {
public:
    virtual ~Checkbox() = default;
    virtual void render() = 0;
    virtual void toggle() = 0;
};

// ================================
// Windows 风格产品
// ================================
class WindowsButton : public Button {
public:
    void render() override {
        std::cout << "[Windows] 渲染方形按钮,蓝色边框\n";
    }
    void onClick() override {
        std::cout << "[Windows] 按钮点击:播放咔哒声\n";
    }
};

class WindowsCheckbox : public Checkbox {
public:
    void render() override {
        std::cout << "[Windows] 渲染方形复选框\n";
    }
    void toggle() override {
        std::cout << "[Windows] 复选框切换状态\n";
    }
};

// ================================
// Mac 风格产品
// ================================
class MacButton : public Button {
public:
    void render() override {
        std::cout << "[Mac] 渲染圆角按钮,灰色渐变\n";
    }
    void onClick() override {
        std::cout << "[Mac] 按钮点击:轻触反馈\n";
    }
};

class MacCheckbox : public Checkbox {
public:
    void render() override {
        std::cout << "[Mac] 渲染圆形复选框\n";
    }
    void toggle() override {
        std::cout << "[Mac] 复选框切换:流畅动画\n";
    }
};

// ================================
// 抽象工厂接口
// ================================
class UIFactory {
public:
    virtual ~UIFactory() = default;
    virtual std::unique_ptr<Button> createButton() = 0;
    virtual std::unique_ptr<Checkbox> createCheckbox() = 0;
};

// ================================
// 具体工厂:生产整套风格
// ================================
class WindowsUIFactory : public UIFactory {
public:
    std::unique_ptr<Button> createButton() override {
        return std::make_unique<WindowsButton>();
    }
    std::unique_ptr<Checkbox> createCheckbox() override {
        return std::make_unique<WindowsCheckbox>();
    }
};

class MacUIFactory : public UIFactory {
public:
    std::unique_ptr<Button> createButton() override {
        return std::make_unique<MacButton>();
    }
    std::unique_ptr<Checkbox> createCheckbox() override {
        return std::make_unique<MacCheckbox>();
    }
};

// ================================
// 客户端:只依赖抽象,不关心具体实现
// ================================
class Application {
private:
    std::unique_ptr<Button> button;
    std::unique_ptr<Checkbox> checkbox;

public:
    Application(UIFactory& factory) {
        button = factory.createButton();
        checkbox = factory.createCheckbox();
    }

    void buildUI() {
        std::cout << "--- 构建界面 ---\n";
        button->render();
        checkbox->render();
    }

    void simulateInteraction() {
        std::cout << "--- 模拟交互 ---\n";
        button->onClick();
        checkbox->toggle();
    }
};

// ================================
// 根据环境选择工厂
// ================================
std::unique_ptr<UIFactory> createFactory(const std::string& os) {
    if (os == "Windows") {
        return std::make_unique<WindowsUIFactory>();
    } else {
        return std::make_unique<MacUIFactory>();
    }
}

int main() {
    std::cout << "======= Windows 应用 =======\n";
    auto winFactory = createFactory("Windows");
    Application winApp(*winFactory);
    winApp.buildUI();
    winApp.simulateInteraction();

    std::cout << "\n======= Mac 应用 =======\n";
    auto macFactory = createFactory("Mac");
    Application macApp(*macFactory);
    macApp.buildUI();
    macApp.simulateInteraction();

    return 0;
}

输出:

复制代码
======= Windows 应用 =======
--- 构建界面 ---
[Windows] 渲染方形按钮,蓝色边框
[Windows] 渲染方形复选框
--- 模拟交互 ---
[Windows] 按钮点击:播放咔哒声
[Windows] 复选框切换状态

======= Mac 应用 =======
--- 构建界面 ---
[Mac] 渲染圆角按钮,灰色渐变
[Mac] 渲染圆形复选框
--- 模拟交互 ---
[Mac] 按钮点击:轻触反馈
[Mac] 复选框切换:流畅动画

优缺点

说明
✅ 优点 保证产品族风格一致,不会"乱配"
✅ 优点 切换产品族只需换一个工厂,代码改动极小
✅ 优点 符合开闭原则(新增产品族只加代码)
❌ 缺点 新增产品种类(比如新增滚动条)需改所有工厂,麻烦
❌ 缺点 类的数量较多,结构复杂

一句话记忆

抽象工厂 = 宜家卖整套家具,北欧风工厂只产北欧货,现代风工厂只产现代货,绝不串货。

相关推荐
Doris_202310 小时前
代码格式化 使用oxfmt
设计模式·架构·前端框架
Doris_202311 小时前
说一说ESLint+Prettier生效的原理
前端·设计模式·架构
Pomelooooo12 小时前
把 git commit 这件事,彻底交给 AI ——一个工程化 /git-commit 命令的设计与落地
设计模式
invicinble13 小时前
设计模式(类的拓扑结构)(描述总纲)
设计模式·原型模式
invicinble16 小时前
设计模式(类的拓扑结构)(为什么会产生设计模式,以及什么是设计模式)
linux·服务器·设计模式
PersonalViolet18 小时前
模板方法模式实战:重构Agent工具审批,告别重复代码
设计模式·agent
老码观察18 小时前
设计模式实战解读(五):策略模式——干掉 if-else 的优雅方案
java·设计模式·策略模式
解决问题no解决代码问题19 小时前
设计模式分类介绍
java·开发语言·设计模式
烬羽19 小时前
从 Python List 到 LLM 接口:一条被忽视的 AI 入门捷径
设计模式
我爱cope20 小时前
【Agent智能体8 | 反思设计模式-大语言模型反思机制的四个演进阶段】
人工智能·设计模式·语言模型