抽象工厂模式 (Abstract Factory Pattern)

抽象工厂模式 (Abstract Factory Pattern) 是一种创建型设计模式,它提供一个创建一系列相关或相互依赖对象的接口,而无需指定它们具体的类。

一、基础

1. 意图

  • 提供一个创建一系列相关或相互依赖对象的接口,而无需指定它们具体的类。

2. 适用场景

  • 当一个系统要独立于它的产品的创建、组合和表示时。
  • 当一个系统要由多个产品系列中的一个来配置时。
  • 当你要强调一系列相关的产品对象的设计以便进行联合使用时。
  • 当你提供一个产品类库,而只想显示它们的接口而不是实现时。

3. 结构

  • 抽象产品族(AbstractProductFamily) :定义了一系列相关产品的抽象接口,这些产品之间存在某种关联或依赖关系。
  • 具体产品(ConcreteProduct) :实现了抽象产品族接口的具体产品类。每个具体产品类对应抽象产品族中的一个产品。
  • 抽象工厂(AbstractFactory) :声明了创建抽象产品族中各个产品的抽象方法。抽象工厂类负责定义创建一系列相关产品的接口,这些产品构成一个产品族。
  • 具体工厂(ConcreteFactory) :实现了抽象工厂中声明的创建各个产品的方法,用于创建具体的产品对象。每个具体工厂类负责创建特定产品族中的所有产品。

二、进阶

1. 抽象工厂模式的优点

  • 抽象工厂模式隔离了具体类的生成,使得客户并不需要知道什么被创建。
  • 当一个产品族中的多个对象被设计成一起工作时,抽象工厂模式能够保证客户端始终只使用同一个产品族中的对象。
  • 增加新的具体工厂和产品族很方便,无需修改已有系统,符合"开闭原则"。

2. 抽象工厂模式的缺点

  • 增加新的产品等级结构麻烦,需要对原有系统进行较大的修改,甚至需要修改抽象层代码,违背了"开闭原则"。

3. 抽象工厂模式的应用

  • 抽象工厂模式广泛应用于各种框架和库中,例如 Spring 框架、Hibernate 框架等。
  • 抽象工厂模式也可以用于开发各种应用程序,例如游戏、图形用户界面等。

三、关键技术

1. 抽象工厂模式的实现

  • 抽象工厂模式可以通过接口或抽象类来实现。
  • 抽象工厂模式可以通过静态方法或实例方法来实现。

2. 抽象工厂模式的扩展

  • 抽象工厂模式可以与其他设计模式结合使用,例如单例模式、原型模式等。
  • 抽象工厂模式可以扩展为工厂方法模式,从而支持创建单一产品对象。

四、易错点

1. 抽象工厂模式的滥用

  • 抽象工厂模式不应该被滥用,只有在需要创建一系列相关或相互依赖的对象时才应该使用抽象工厂模式。

2. 抽象工厂模式的性能问题

  • 抽象工厂模式可能会降低代码的性能,因为每次创建对象时都需要调用工厂方法。

五、核心代码

5.1 抽象工厂模式的简单实现

cpp 复制代码
// AbstractProductA
class AbstractProductA {
public:
    virtual void use() = 0;
};

// ConcreteProductA1
class ConcreteProductA1 : public AbstractProductA {
public:
    void use() override {
        std::cout << "Using ConcreteProductA1" << std::endl;
    }
};

// ConcreteProductA2
class ConcreteProductA2 : public AbstractProductA {
public:
    void use() override {
        std::cout << "Using ConcreteProductA2" << std::endl;
    }
};

// AbstractProductB
class AbstractProductB {
public:
    virtual void eat() = 0;
};

// ConcreteProductB1
class ConcreteProductB1 : public AbstractProductB {
public:
    void eat() override {
        std::cout << "Eating ConcreteProductB1" << std::endl;
    }
};

// ConcreteProductB2
class ConcreteProductB2 : public AbstractProductB {
public:
    void eat() override {
        std::cout << "Eating ConcreteProductB2" << std::endl;
    }
};

// AbstractFactory
class AbstractFactory {
public:
    virtual AbstractProductA* createProductA() = 0;
    virtual AbstractProductB* createProductB() = 0;
};

// ConcreteFactory1
class ConcreteFactory1 : public AbstractFactory {
public:
    AbstractProductA* createProductA() override {
        return new ConcreteProductA1();
    }

    AbstractProductB* createProductB() override {
        return new ConcreteProductB1();
    }
};

// ConcreteFactory2
class ConcreteFactory2 : public AbstractFactory {
public:
    AbstractProductA* createProductA() override {
        return new ConcreteProductA2();
    }

    AbstractProductB* createProductB() override {
        return new ConcreteProductB2();
    }
};

// Client
int main() {
    AbstractFactory* factory = new ConcreteFactory1();
    AbstractProductA* productA = factory->createProductA();
    AbstractProductB* productB = factory->createProductB();

    productA->use();
    productB->eat();

    delete factory;
    delete productA;
    delete productB;

    factory = new ConcreteFactory2();
    productA = factory->createProductA();
    productB = factory->createProductB();

    productA->use();
    productB->eat();

    delete factory;
    delete productA;
    delete productB;

    return 0;
}

5.2 抽象工厂模式的扩展实现

cpp 复制代码
// AbstractProductA
class AbstractProductA {
public:
    virtual void use() = 0;
};

// ConcreteProductA1
class ConcreteProductA1 : public AbstractProductA {
public:
    void use() override {
        std::cout << "Using ConcreteProductA1" << std::endl;
    }
};

// ConcreteProductA2
class ConcreteProductA2 : public AbstractProductA {
public:
    void use() override {
        std::cout << "Using ConcreteProductA2" << std::endl;
    }
};

// AbstractProductB
class AbstractProductB {
public:
    virtual void eat() = 0;
};

// ConcreteProductB1
class ConcreteProductB1 : public AbstractProductB {
public:
    void eat() override {
        std::cout << "Eating ConcreteProductB1" << std::endl;
    }
};

// ConcreteProductB2
class ConcreteProductB2 : public AbstractProductB {
public:
    void eat() override {
        std::cout << "Eating ConcreteProductB2" << std::endl;
    }
};

// AbstractFactory
class AbstractFactory {
public:
    virtual AbstractProductA* createProductA() = 0;
    virtual AbstractProductB* createProductB() = 0;
};

// ConcreteFactory1
class ConcreteFactory1 : public AbstractFactory {
public:
    AbstractProductA* createProductA() override {
        return new ConcreteProductA1();
    }

    AbstractProductB* createProductB() override {
        return new ConcreteProductB1();
    }
};

// ConcreteFactory2
class ConcreteFactory2 : public AbstractFactory {
public:
    AbstractProductA* createProductA() override {
        return new ConcreteProductA2();
    }

    AbstractProductB* createProductB() override {
        return new ConcreteProductB2();
    }
};

// Client
int main() {
    AbstractFactory* factory = new ConcreteFactory1();
    AbstractProductA* productA = factory->createProductA();
    AbstractProductB* productB = factory->createProductB();

    productA->use();
    productB->eat();

    delete factory;
    delete productA;
    delete productB;

    factory = new ConcreteFactory2();
    productA = factory->createProductA();
    productB = factory->createProductB();

    productA->use();
    productB->eat();

    delete factory;
    delete productA;
    delete productB;

    return 0;
}

六、总结

抽象工厂模式通过将对象的创建和使用进一步分离,为创建一系列相关或相互依赖的对象提供了一种优雅的解决方案。它在提高系统的可扩展性和可维护性方面具有显著优势,但同时也增加了系统的复杂性。在实际开发中,需要根据具体的业务需求和系统架构来权衡是否使用抽象工厂模式。如果系统中存在多个产品族,并且产品之间存在复杂的依赖关系,那么抽象工厂模式是一个很好的选择。

相关推荐
Small black human1 小时前
设计模式-应用分层
设计模式
OneQ6662 小时前
C++讲解---创建日期类
开发语言·c++·算法
Coding小公仔4 小时前
C++ bitset 模板类
开发语言·c++
菜鸟看点5 小时前
自定义Cereal XML输出容器节点
c++·qt
悲伤小伞6 小时前
linux_git的使用
linux·c语言·c++·git
ysa0510306 小时前
数论基础知识和模板
数据结构·c++·笔记·算法
小小小小王王王9 小时前
求猪肉价格最大值
数据结构·c++·算法
岁忧10 小时前
(LeetCode 面试经典 150 题 ) 58. 最后一个单词的长度 (字符串)
java·c++·算法·leetcode·面试·go
码农秋10 小时前
设计模式系列(10):结构型模式 - 桥接模式(Bridge)
设计模式·桥接模式
GodKeyNet10 小时前
设计模式-桥接模式
java·设计模式·桥接模式