C++设计模式之抽象工厂模式
_学习GeekBand李建忠老师课程记录
1.动机:
1.在软件系统中,经常面临着“一系列相互依赖的对象”的创建工作;同时,由于需求的变化,往往存在更多系列对象的创建工作。
如Sql的Connection Command Reader与 Oracle的 Connection Command Reader的一系列同类型的 相关联性的创建工作。
2.如何应对这种变化?如何绕过常规的对象创建方法(new),提供一种“封装机制”来避免客户程序和这种“多系列具体对象创建工作”的紧耦合?
2.模式定义:
提供一个接口,让该接口负责创建一系列相关或者相互依赖的对象,无需指定他们具体的类。-GoF
3.抽象工厂重构前代码:
class EmployeeDAO {抽象工厂重构前代码
public:
vector<EmployeeDAO> GetEmployees() {
SqlConnection* connection= new SqlConnection();
connection->ConnectionString("...");//创建SQL数据库链接
sqlCommand* command=new sqlCommand();
command->CommandText("...");
command->SqlConnection(connection);//链接数据库
SqlDataReader* reader=command->ExecuteReader();
while(reader->Read()){//启动数据库读写
}
}
};
4.重构后代码1 简单工厂模式:
//数据库基类
/******************************************************************/
class IDBConnection{
};
class IDBConnectionFactory{//工厂1 工厂生成Connection
public:
virtual IDBConnection* CreateDBConnection()=0;
};
/******************************************************************/
class IDBCommand{
};
class IDBCommandFactory{//工厂2 工厂生成Command
public:
virtual IDBCommand* CreateDBCommand()=0;
};
/******************************************************************/
class IDataReader {
};
class IDataReaderFactory{//工厂3 工厂生成读写reader
public:
virtual IDataReader* CreateDataReader()=0;
};
/******************************************************************/
支持SQL Server
class SqlConnection:public IDBConnection {
};
class SqlConnectionFactory:public IDBConnectionFactory {
};
class SqlCommand:public IDBCommand {
};
class SqlCommandFactory:public IDBConnectionFactory {
};
class SqlDataReader:public IDDataReader {
};
class SqlDataReaderFactory:public IDataReaderFactory{
};
/******************************************************************/
//支持Oracle
class OracleConnection: public IDBConnection{
};
class OracleConnectionFactory:public IDBConnectionFactory {
};
class OracleCommand: public IDBCommand{
};
class OracleCommandFactory:public IDBConnectionFactory {
};
class OracleDataReader: public IDataReader{
};
class OracleDataReaderFactory:public IDataReaderFactory{
};
/******************************************************************/
class EmployeeDAO {
private:
IDBConnectionFactory* dbconnectionFactory;//三个指针有关联性 都是同一种的数据库类型
IDBCommandFactory* dbcommandfactory;//如果三个指针不同 将会报错
IDataReaderFactory* datareaderfactory;
public:
vector<EmployeeDAO> GetEmployees() {
IDBConnection* connection= dbconnectionFactory->CreateDBConnection();
connection->ConnectionString("...");
IDBCommand* command=dbcommandfactory->CreateDBCommand();
command->CommandText("...");
command->SetConnection(connection);
IDataReader* reader=command->ExecuteReader();
while(reader->Read()) {
}
}
};
5. 重构代码2 抽象工厂 Abstract Factory :
//数据库访问有关的基类
class IDBConnection{
};
class IDBCommand{
};
class IDataReader{
};
//工厂的抽象基类
class IDBFactory{
public:
virtual IDBConnection* CreateDBConnection()=0;
virtual IDBCommand* CreateDBCommand()=0;
virtual IDataReader* CreateDataReader()=0;
};
/*****************************************************************************************/
//支持SQL Server
class SqlConnection: public IDBConnection{
};
class SqlCommand: public IDBCommand{
};
class SqlDataReader: public IDataReader{
};
//Sql的工厂
class SqlDBFactory:public IDBFactory {
public:
virtual IDBConnection* CreateDBConnection(){};
virtual IDBCommand* CreateDBCommand(){};
virtual IDataReader* CreateDataReader(){};
}
/*****************************************************************************************/
//支持Oracle
class OracleConnection: public IDBConnection{
};
class OracleCommand: public IDBCommand{
};
class OracleDataReader: public IDataReader{
};
//Oracle的工厂
class OracleDBFactory:public IDBFactory {
public:
virtual IDBConnection* CreateDBConnection(){};
virtual IDBCommand* CreateDBCommand(){};
virtual IDataReader* CreateDataReader(){};
}
/*****************************************************************************************/
class EmployeeDAO{
IDBFactory* m_dbfactory;//定义一个抽象工厂的指针
public:
vector<EmployeeDAO> GetEmployees() {
IDBConnection* connection=m_dbfactory->CreateDBConnection();
connection->ConnectionString("...");
IDBCommand* command=m_dbfactory->CreateDBCommand();
command->CommandText("...");
command->SetConnection(connection);//关联性
IDDataReader* reader=command->ExecuteReader();
while(reader->Read()) {
//...........
}
}
};
6. 结构图:
7. 要点总结:
1.如果没有应对“多系列对象构建”的需求变化,则没有必要使用Abstract Factory模式,这时候使用简单的工厂完全可以。
2.如果没有应对“多系列对象构建”的需求变化,则没有必要使用Abstract Factory模式,这时候使用简单的工厂完全可以。
3.Abstract Factory模式主要在于应对“新系列”的需求变动。其缺点在于难以应对“新对象”的需求变动。