一、概念

  提供一个接口,让该接口负责创建一系列“相关或相互依赖的对象”,无需指定它们具体的类。 —《设计模式》Gof

二、动机

  在软件系统中,经常面临着“一系列相互依赖的对象”的创建工作;同时,由于需求的变化,往往存在更多系列对象的创建工作

  如何应对这种变化?如何绕过常规的对象创建方法(new),提供一种“封装机制”来避免客户程序和这种“具体对象创建工作”的紧耦合。

三、源代码讲解

class EmployeeDAO{
    
public:
    vector<EmployeeDO> GetEmployees(){
        SqlConnection* connection =  //数据库连接
            new SqlConnection();
        connection->ConnectionString = "...";

        SqlCommand* command =  //数据库命令
            new SqlCommand();
        command->CommandText="...";
        command->SetConnection(connection);

        SqlDataReader* reader = command->ExecuteReader(); //数据库信息读取
        while (reader->Read()){

        }
    }
};

四、使用抽象工厂模式进行改进

// 数据库访问有关的基类
class IDBConnection {
    
};

class IDBCommand {
    
};

class IDataReader {
    
};

// 支持SQL Server(扩展)
class SqlConnection: public IDBConnection {
    
};
class SqlCommand: public IDBCommand {
    
};
class SqlDataReader: public IDataReader {
    
};

// 支持Oracle(扩展)
class OracleConnection: public IDBConnection {
    
};

class OracleCommand: public IDBCommand {
    
};

class OracleDataReader: public IDataReader {
    
};

// 抽象工厂(应对“多系列对象构建”的需求变化)
class IDBFactory {
public:
	// 高内聚
    virtual IDBConnection* CreateDBConnection() = 0;
    virtual IDBCommand* CreateDBCommand() = 0;
    virtual IDataReader* CreateDataReader() = 0;
};

// 具体的工厂
class SqlDBFactory : public IDBFactory {
public:
    virtual IDBConnection* CreateDBConnection() = 0;
    virtual IDBCommand* CreateDBCommand() = 0;
    virtual IDataReader* CreateDataReader() = 0;
};

class OracleDBFactory : public IDBFactory {
public:
    virtual IDBConnection* CreateDBConnection() = 0;
    virtual IDBCommand* CreateDBCommand() = 0;
    virtual IDataReader* CreateDataReader() = 0;
};

class EmployeeDAO {
public:
    vector<EmployeeDO> GetEmployees() {
        IDBConnection* connection = dbFactory->CreateDBConnection();
        connection->ConnectionString("...");

        IDBCommand* command = dbFactory->CreateDBCommand();
        command->CommandText("...");
        command->SetConnection(connection); // 关联性

        IDBDataReader* reader = command->ExecuteReader(); // 关联性
        while (reader->Read()) {

        }
    }
private:
	IDBFactory* dbFactory;
};

五、类图结构

在这里插入图片描述

六、要点总结

  • 如果没有应对“多系列对象构建”的需求变化,则没有必要使用 Abstract Factory 模式,这时候使用简单的工厂完全可以。

  • “系列对象”指的是在某一特定系列下的对象之间具有相互依赖或作用的关系,不同系列的对象之间不能相互依赖。

  • Abstract Factory 模式主要在于应对“新系列”的需求变动,其缺点在于难以应对“新对象”的需求变动

Logo

开放原子开发者工作坊旨在鼓励更多人参与开源活动,与志同道合的开发者们相互交流开发经验、分享开发心得、获取前沿技术趋势。工作坊有多种形式的开发者活动,如meetup、训练营等,主打技术交流,干货满满,真诚地邀请各位开发者共同参与!

更多推荐