一:概念

抽象工厂模式是所有形态的工厂模式中最为抽象和最具一般性的。
抽象工厂模式可以向客户端提供一个接口,使得客户端在不必指定产品的具体类型的情况下,能够创建多个产品族的产品对象

二:动机

在软件系统中,经常面临着“一系列相互依赖的对象”的创建工作;同时,由于需求的变化,往往存在更多系列对象的创建工作。
如何应对这种变化?如何绕过常规的对象创建方法(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()){ }
}
};

问题提出:

需求更改:需要变更数据库mysql,oracle,sqlite等
所以new是不合适的,是静态特质,定死了。这个类就不适用于多种数据库变化的可能

(二)支持面向接口编程

//数据库访问有关的基类
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 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()){ } }
};
从上篇文章可以知道new是不好的,需要编译时依赖。所以我们想办法使用工厂模式修改去掉new

(三)添加工厂

//数据库访问有关的基类
class IDBConnection{ };
class IDBConnectionFactory{
public:
virtual IDBConnection* CreateDBConnection()=;
}; class IDBCommand{ };
class IDBCommandFactory{
public:
virtual IDBCommand* CreateDBCommand()=;
}; class IDataReader{ };
class IDataReaderFactory{
public:
virtual IDataReader* CreateDataReader()=;
}; //支持SQL Server
class SqlConnection: public IDBConnection{ };
class SqlConnectionFactory:public IDBConnectionFactory{ }; class SqlCommand: public IDBCommand{ };
class SqlCommandFactory:public IDBCommandFactory{ }; class SqlDataReader: public IDataReader{ };
class SqlDataReaderFactory:public IDataReaderFactory{ }; //支持Oracle
class OracleConnection: public IDBConnection{ }; class OracleCommand: public IDBCommand{ }; class OracleDataReader: public IDataReader{ };

//.....也有Oracle相关工厂 class EmployeeDAO{
   //根据下面抽象基类,我们可以来创建SQL,Oracle等多种数据库
IDBConnectionFactory* dbConnectionFactory;
IDBCommandFactory* dbCommandFactory;
IDataReaderFactory* dataReaderFactory;
public:
vector<EmployeeDO> GetEmployees(){
IDBConnection* connection =
dbConnectionFactory->CreateDBConnection();
connection->ConnectionString("..."); IDBCommand* command =
dbCommandFactory->CreateDBCommand();
command->CommandText("...");
command->SetConnection(connection); //关联性 IDBDataReader* reader = command->ExecuteReader(); //关联性
while (reader->Read()){ }
}
};
勉强解决了变更数据库的问题

新的问题:

    IDBConnectionFactory* dbConnectionFactory;
IDBCommandFactory* dbCommandFactory;
IDataReaderFactory* dataReaderFactory;
我们若是传入3个不同的变量,mysql的连接,Oracle的命令,sqlite的读取,那么就会报错,紊乱,因为原来这三个是有关联的
        IDBConnection* connection =
dbConnectionFactory->CreateDBConnection();
connection->ConnectionString("..."); IDBCommand* command =
dbCommandFactory->CreateDBCommand();
command->CommandText("...");
command->SetConnection(connection); //关联性 IDBDataReader* reader = command->ExecuteReader(); //关联性
什么样的数据库就和什么样的命令相关联,数据库连接对于每个数据库也是不一样的,所以,我们需要要保证关联性一致

(四)引出抽象工厂

//数据库访问有关的基类
class IDBConnection{ }; class IDBCommand{ }; class IDataReader{ }; class IDBConnectionFactory{
public:
virtual IDBConnection* CreateDBConnection()=;
}; class IDBCommandFactory{
public:
virtual IDBCommand* CreateDBCommand()=;
}; class IDataReaderFactory{
public:
virtual IDataReader* CreateDataReader()=;
};
我们发现3个类的相关性很强,那么我们就可以使用一个工厂来实现:高内聚
//数据库访问有关的基类
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{ }; class SqlDBFactory:public IDBFactory{
public:
virtual IDBConnection* CreateDBConnection()=0;
virtual IDBCommand* CreateDBCommand()=0;
virtual IDataReader* CreateDataReader()=0
; };
//支持Oracle
class OracleConnection: public IDBConnection{ }; class OracleCommand: public IDBCommand{ }; class OracleDataReader: public IDataReader{ }; class OracleDBFactory:public IDBFactory{
public:
virtual IDBConnection* CreateDBConnection()=0;
virtual IDBCommand* CreateDBCommand()=0;
virtual IDataReader* CreateDataReader()=0
;
};
class EmployeeDAO{
IDBFactory* dbFactory; 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()){ }
}
};

五:模式定义

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

                                                                             --《设计模式》Gof

六:类图(结构)

其中还少了一个抽象产品IDataReader,就可以和上面代码对应上了。

七:要点总结

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

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

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

新系列:是指mysql,Oracle,sqlite等各是一个系列
新对象的需求变化:像是连接,命令,查询,我们若是再添加一个变动到抽象基类,可以往往不适用全部的系列(抽象基类要求稳定)

八:案例实现(南北水果族)

class Fruit
{
public:
virtual void sayname() = ;
virtual ~Fruit(){}
}; class FruitFactory
{
public:
virtual Fruit* getApple() = ;
virtual Fruit* getBanana() = ;
virtual ~FruitFactory(){};
};
class NorthApple :public Fruit
{
public:
virtual void sayname()
{
cout << "you get a north apple" << endl;
}
}; class NorthBanana :public Fruit
{
public:
virtual void sayname()
{
cout << "you get a north banana" << endl;
}
}; class NorthFruitFactory :public FruitFactory
{
public:
virtual Fruit* getApple()
{
return new NorthApple();
} virtual Fruit* getBanana()
{
return new NorthBanana();
}
};
class SouthApple :public Fruit
{
public:
virtual void sayname()
{
cout << "you get a south apple" << endl;
}
}; class SouthBanana :public Fruit
{
public:
virtual void sayname()
{
cout << "you get a south banana" << endl;
}
}; class SouthFruitFactory :public FruitFactory
{
public:
virtual Fruit* getApple()
{
return new SouthApple();
} virtual Fruit* getBanana()
{
return new SouthBanana();
}
};
void main()
{
FruitFactory *ff = new NorthFruitFactory();
Fruit* ap = ff->getApple();
Fruit* ba = ff->getBanana();
ap->sayname();
ba->sayname();
system("pause");
return;
}

设计模式---对象创建模式之抽象工厂模式(Abstract Factory)的更多相关文章

  1. Java设计模式(3)——创建型模式之抽象工厂模式(Abstract Factory)

    一.概述 抽象工厂模式是指当有多个抽象角色时,使用的一种工厂模式.抽象工厂模式可以向客户端提供一个接口,使客户端在不必指定产品的具体情况下,创建多个产品族中的产品对象. UML图: 其他的过多概念不再 ...

  2. java设计模式 -------- 创建模式 之 抽象工厂模式

    本文是自己学习所做笔记,欢迎转载,但请注明出处:http://blog.csdn.net/jesson20121020 工厂方法模式和抽象工厂模式不好区分清楚: 工厂方法模式特点: 1. 一个抽象产品 ...

  3. 设计模式的征途—3.抽象工厂(Abstract Factory)模式

    上一篇的工厂方法模式引入了工厂等级结构,解决了在原来简单工厂模式中工厂类职责太重的原则,但是由于工厂方法模式的每个工厂只生产一类产品,可能会导致系统中存在大量的工厂类,从而增加系统开销.那么,我们应该 ...

  4. 设计模式的征途—4.抽象工厂(Abstract Factory)模式

    上一篇的工厂方法模式引入了工厂等级结构,解决了在原来简单工厂模式中工厂类职责太重的原则,但是由于工厂方法模式的每个工厂只生产一类产品,可能会导致系统中存在大量的工厂类,从而增加系统开销.那么,我们应该 ...

  5. Java设计模式之【工厂模式】(简单工厂模式,工厂方法模式,抽象工厂模式)

    Java设计模式之[工厂模式](简单工厂模式,工厂方法模式,抽象工厂模式) 工厂模式出现的原因 在java中,创建一个对象最简单的方法就是使用new关键字.但在一些复杂的业务逻辑中,创建一个对象不只需 ...

  6. iOS常用设计模式——工厂方法(简单工厂模式,工厂方法模式, 抽象工厂模式)

    1. 简单工厂模式 如何理解简单工厂,工厂方法, 抽象工厂三种设计模式? 简单工厂方法包含:父类拥有共同基础接口,具体子类实现子类特殊功能,工厂类根据参数区分创建不同子类实例.该场景对应的UML图如下 ...

  7. 设计模式之第2章-抽象工厂模式(Java实现)

    设计模式之第2章-抽象工厂模式(Java实现) “上次是我的不对,贿赂作者让我先讲来着,不过老婆大人大人有大量,不与我计较,这次还让我先把上次未讲完的应用场景部分给补充上去,有妻如此,夫复何求.”(说 ...

  8. iOS经常使用设计模式——工厂方法(简单工厂模式,工厂方法模式, 抽象工厂模式)

    1. 简单工厂模式 怎样理解简单工厂,工厂方法. 抽象工厂三种设计模式? 简单工厂的生活场景.卖早点的小摊贩.他给你提供包子,馒头,地沟油烙的煎饼等,小贩是一个工厂.它生产包子,馒头,地沟油烙的煎饼. ...

  9. python 设计模式之工厂模式 Factory Pattern (简单工厂模式,工厂方法模式,抽象工厂模式)

    十一回了趟老家,十一前工作一大堆忙成了狗,十一回来后又积累了一大堆又 忙成了狗,今天刚好抽了一点空开始写工厂方法模式 我看了<Head First 设计模式>P109--P133 这25页 ...

  10. C#设计模式学习笔记:(3)抽象工厂模式

    本笔记摘抄自:https://www.cnblogs.com/PatrickLiu/p/7596897.html,记录一下学习过程以备后续查用. 一.引言 接上一篇C#设计模式学习笔记:简单工厂模式( ...

随机推荐

  1. const修饰符与函数

    一.用const修饰函数的参数 函数参数类型前加const指明该参数为常量,在函数内部不可改变. void func(const int x) { //x不可以在内部进行赋值等操作. } 注:当参数为 ...

  2. number (2)变量相关错误

    变量没有被定义 fw cannot be resolved 变量没有被初始化 正确代码 package com.itheima_01; import java.io.FileWriter;import ...

  3. 美化centos7

    在美化前,我们先安装一个扩展源.yum install -y epel-release然后安装字体包yum -y install liberation-mono-fonts 安装gnome-menis ...

  4. centos7改中文

    centos7的与centos6有少许不同: 1.安装中文包: root@iZj6cbstl2n6r280a27eppZ tmp]# yum groupinstall "fonts" ...

  5. Hbase Shell 数据操作说明

    启动.关闭hbase./bin/start-hbase.sh./bin/stop-hbase.sh 查询HBase版本.它的语法如下:hbase(main):010:0> version 建表c ...

  6. mysql DDL&DML 语言

    DDL:数据定义语言 CREATE, ALTER, DROP CREATE相关的常用命令: CREATE DATABASECREATE EVENTCREATE FUNCTIONCREATE FUNCT ...

  7. ansible系列2-常用命令

    copyansible oldboy -m copy -a "src=/etc/hosts dest=/tmp/ mode=0600 owner=oldboy group=oldboy &q ...

  8. BZOJ3669[Noi2014]魔法森林——kruskal+LCT

    题目描述 为了得到书法大家的真传,小E同学下定决心去拜访住在魔法森林中的隐士.魔法森林可以被看成一个包含个N节点M条边的无向图,节点标号为1..N,边标号为1..M.初始时小E同学在号节点1,隐士则住 ...

  9. 在finally块中使用try catch,并且catch的时候抛出异常的一个问题

    在finally中使用try/catch,并且catch的时候抛出异常 IDEA会提示警告 Reports throw statements inside of finally blocks. Whi ...

  10. day26 多继承

    class A(object): def test(self): print('from A') class B(A): def test(self): print('from B') class C ...