问题描述

在面向对象系统设计中经常可以遇到以下的两类问题:
1)为了提高内聚(Cohesion)和松耦合(Coupling),我们经常会抽象出一些类的公共接口以形成抽象基类或者接口。这样我们可以通过声明一个指向基类的指针来指向实际的子类实现, 达到了多态的目的。

这里很容易出现的一个问题 n 多的子类继承自抽象基类, 我们不得不在每次要用到子类的地方就编写诸如 new ×××;的代码。这里带来两个问题:

->1.客户程序员必须知道实际子类的名称(当系统复杂后, 命名将是一个很不好处理的问题, 为了处理可能的名字冲突, 有的命名可能并不是具有很好的可读性和可记忆性, 就姑且不论不同程序员千奇百怪的个人偏好了。)

->2.程序的扩展性和维护变得越来越困难。

2)还有一种情况就是在父类中并不知道具体要实例化哪一个具体的子类。

假设我们在类 A 中要使用到类 B, B 是一个抽象父类,在 A 中并不知道具体要实例化那一个 B 的子类,但是在类A的子类D中是可以知道的。在A中我们没有办法直接使用类似于new×××的语句,因为根本就不知道×××是什么。

以上两个问题也就引出了 Factory 模式的两个最重要的功能:
1)定义创建对象的接口,封装了对象的创建;
2)使得具体化类的工作延迟到了子类中。

在第一个问题中,我们经常就是声明一个创建对象的接口,并封装了对象的创建过程。 Factory 这里类似于一个真正意义上的工厂(生产对象)。

在第二个问题中,我们需要提供一个对象创建对象的接口,并在子类中提供其具体实现(因为只有在子类中可以决定到底实例化哪一个)。

UML类图

对于工厂模式,具体上可以分为三类:

  1. 简单工厂模式;
  2. 工厂方法模式;
  3. 抽象工厂模式。

对于上面的三种工厂模式,从上到下逐步抽象,并且更具一般性。本篇主要论述第一类(简单工厂模式)和第二类(工厂方法模式)。

第一种情况下(对应的就是“简单工厂模式”):

上图所示的Factory模式经常在系统开发中用到,但是这并不是 Factory模式的最大威力所在(因为这可以通过其他方式解决这个问题)。Factory模式不单是提供了创建对象的接口,其最重要的是延迟了子类的实例化(第二个问题)。

如下图所示(对应的就是工厂方法模式):

这种模式的应用并不是只是为了封装对象的创建,而是要把对象的创建放到子类中实现: Factory 中只是提供了对象创建的接口,其实现将放在 Factory 的子类ConcreteFactory 中进行。

适用场合

简单工厂模式:

  1. 在程序中,需要创建的对象很多,导致对象的new操作多且杂时,需要使用简单工厂模式;
  2. 由于对象的创建过程是我们不需要去关心的,而我们注重的是对象的实际操作,所以,我们需要分离对象的创建和操作两部分,如此,方便后期的程序扩展和维护。

工厂方法模式:

工厂方法模式的意义是定义一个创建产品对象的工厂接口,将实际创建工作推迟到子类当中。它修正了简单工厂模式中不遵守开放—封闭原则。核心工厂类不再负责产品的创建,这样核心类成为一个抽象工厂角色,仅负责具体工厂子类必须实现的接口,这样进一步抽象化的好处是使得工厂方法模式可以使系统在不修改具体工厂角色的情况下引进新的产品。

  1. 在设计的初期,就考虑到产品在后期会进行扩展的情况下,可以使用工厂方法模式;
  2. 产品结构较复杂的情况下,可以使用工厂方法模式;

由于使用设计模式是在详细设计时,就需要进行定夺的,所以,需要权衡多方面的因素,而不能为了使用设计模式而使用设计模式。

代码实现

简单工厂模式:

#include <iostream>
#include <string>
using namespace std;
//基类
class Operation
{
public:
double numberA, numberB;
virtual double getResult()
{
return 0;
}
};
//加法
class addOperation :public Operation
{
double getResult()
{
return numberA + numberB;
}
}; //减法
class subOperation :public Operation
{
double getResult()
{
return numberA - numberB;
}
};
//乘法
class mulOperation :public Operation
{
double getResult()
{
return numberA*numberB;
}
};
//除法
class divOperation :public Operation
{
double getResult()
{
return numberA / numberB;
}
};
//工厂类
class operFactory
{
public:
static Operation *createOperation(char c)
{
//在c#中可以用反射来取消判断时用的switch,那么c++中用的是啥呢?RTTI?
switch (c)
{
case '+':
return new addOperation;
break; case '-':
return new subOperation;
break; case '*':
return new mulOperation;
break; case '/':
return new divOperation;
break;
}
}
}; int main()
{
Operation *oper = operFactory::createOperation('+');
oper->numberA = 9;
oper->numberB = 99;
cout << oper->getResult() << endl;
return 0;
}

运行结果:

工厂方法模式:

#include <iostream>
#include <string>
using namespace std; class Operation
{
public:
double numberA, numberB;
virtual double getResult()
{
return 0;
}
}; class addOperation :public Operation
{
double getResult()
{
return numberA + numberB;
}
}; class subOperation :public Operation
{
double getResult()
{
return numberA - numberB;
}
}; class mulOperation :public Operation
{
double getResult()
{
return numberA*numberB;
}
}; class divOperation :public Operation
{
double getResult()
{
return numberA / numberB;
}
}; class IFactory
{
public:
virtual Operation *createOperation() = 0;
}; class AddFactory :public IFactory
{
public:
static Operation *createOperation()
{
return new addOperation();
}
}; class SubFactory :public IFactory
{
public:
static Operation *createOperation()
{
return new subOperation();
}
}; class MulFactory :public IFactory
{
public:
static Operation *createOperation()
{
return new mulOperation();
}
}; class DivFactory :public IFactory
{
public:
static Operation *createOperation()
{
return new divOperation();
}
}; int main()
{
Operation *oper = MulFactory::createOperation();
oper->numberA = 9;
oper->numberB = 99;
cout << oper->getResult() << endl;
return 0;
}

运行结果:

---------------------------------------------------------

#include <iostream>
#include <string>
using namespace std;
//实例基类,相当于Product(为了方便,没用抽象)
class LeiFeng
{
public:
virtual void sweep()
{
cout << "雷锋扫地" << endl;
} void wash()
{
cout << "雷锋洗衣服" << endl;
}
};
//学雷锋的大学生,相当于ConcreteProduct
class Student :public LeiFeng
{
public:
virtual void sweep(){
cout << "大学生扫地" << endl;
}
};
//学雷锋的志愿者,相当于ConcreteProduct
class Volunteer :public LeiFeng
{
public:
virtual void sweep(){
cout << "zhiyaunzhe" << endl;
}
}; //工厂基类Creator
class LeiFengFactory
{
public:
virtual LeiFeng *createLeiFeng()
{
return new LeiFeng();
}
}; //工厂具体类
class StudentFactory :public LeiFengFactory
{
public:
virtual LeiFeng *createLeiFeng()
{
return new Student();
}
}; class VolFactory :public LeiFengFactory
{
public:
virtual LeiFeng *createLeiFeng()
{
return new Volunteer();
}
}; int main()
{
LeiFengFactory *sf = new LeiFengFactory();
LeiFeng *s = sf->createLeiFeng();
s->sweep();
delete s;
delete sf;
return 0;
}

运行结果:

  

参考文献:  

《大话设计模式 C++》

《C++设计模式》

另外可参考博客(其系列写得不错):C++设计模式——简单工厂模式

设计模式之工厂模式(c++)的更多相关文章

  1. 设计模式——抽象工厂模式及java实现

    设计模式--抽象工厂模式及java实现 设计模式在大型软件工程中很重要,软件工程中采用了优秀的设计模式有利于代码维护,方便日后更改和添加功能. 设计模式有很多,而且也随着时间在不断增多,其中最著名的是 ...

  2. 5. 星际争霸之php设计模式--抽象工厂模式

    题记==============================================================================本php设计模式专辑来源于博客(jymo ...

  3. 3. 星际争霸之php设计模式--简单工厂模式

    题记==============================================================================本php设计模式专辑来源于博客(jymo ...

  4. iOS 设计模式之工厂模式

    iOS 设计模式之工厂模式 分类: 设计模式2014-02-10 18:05 11020人阅读 评论(2) 收藏 举报 ios设计模式 工厂模式我的理解是:他就是为了创建对象的 创建对象的时候,我们一 ...

  5. 设计模式之工厂模式(Factory)

    设计模式的工厂模式一共有三种:简单工厂模式,工厂模式,抽象工厂模式 简单工厂模式原理:只有一个工厂类,通过传参的形式确定所创建的产品对象种类 代码如下: #include <stdio.h> ...

  6. php设计模式:工厂模式

    php设计模式:工厂模式 意图: 定义一个用于创建对象的接口,让子类决定实例化哪一个类. 工厂模式实现: 工厂模式中任何创建对象的工厂类都要实现这个接口,实现接口的方法体中都要实现接口中的方法,它声明 ...

  7. 浅析JAVA设计模式之工厂模式(一)

    1 工厂模式简单介绍 工厂模式的定义:简单地说,用来实例化对象,取代new操作. 工厂模式专门负责将大量有共同接口的类实例化.工作模式能够动态决定将哪一个类实例化.不用先知道每次要实例化哪一个类. 工 ...

  8. java 设计模式之工厂模式与反射的结合

    工厂模式: /**  * @author Rollen-Holt 设计模式之 工厂模式  */   interface fruit{     public abstract void eat(); } ...

  9. C#学习之设计模式:工厂模式

    最近研究一下设计模式中工厂模式的应用,在此记录如下: 什么是工厂模式? 工厂模式属于设计模式中的创造型设计模式的一种.它的主要作用是协助我们创建对象,为创建对象提供最佳的方式.减少代码中的耦合程度,方 ...

  10. [JS设计模式]:工厂模式(3)

    简单工厂模式是由一个方法来决定到底要创建哪个类的实例, 而这些实例经常都拥有相同的接口. 这种模式主要用在所实例化的类型在编译期并不能确定, 而是在执行期决定的情况. 说的通俗点,就像公司茶水间的饮料 ...

随机推荐

  1. 20145203盖泽双《网络对抗技术》实践五:MSF基础应用

    20145203盖泽双<网络对抗技术>实践五:MSF基础应用 1.实践目标 掌握metasploit的基本应用方式,掌握常用的三种攻击方式的思路.下面是我自己做的时候用的四个套路. (1) ...

  2. gulp学习-metamask前端使用

    https://www.gulpjs.com.cn/docs/getting-started/ ,这个是3.9.0版本 后面发现安装的版本是4.0.0,看下面这个: https://github.co ...

  3. linux grep 的使用

    常用的 grep 选项有:    -c 只输出匹配行的个数.    -i 不区分大小写(只适用于单字符).    -h 查询多文件时不显示文件名.    -l 查询多文件时只输出包含匹配字符的文件名. ...

  4. 开发工具|给你的项目买份保险:Python虚拟环境

    读完需要 9 分钟 1. 什么是虚拟环境? 虚拟环境的意义,就如同 虚拟机 一样,它可以实现不同环境中Python依赖包相互独立,互不干扰.这在一定程度的意义上,给了我们的项目一份很有力的保障.在这里 ...

  5. ASP.NET MVC学习笔记(一) 从路由开始创建mvc

    之前一篇写一半发现版本太老了,是基于mvc2的. 两本参考书编写的顺序各方面都不太一样.决定重新写一篇. 我这篇文章基于mvc5,vs2015 参考书:Will保哥的ASP.NET MVC4开发指南 ...

  6. SpringSecurity初步理解

    Authenticating a User with LDAP 首先创建一个简单的web控制器 package hello; import org.springframework.web.bind.a ...

  7. Django学习篇(第二部)

    4.Django pip3 install django C:\Python35\Scripts # 创建Django工程 django-admin startproject [工程名称] mysit ...

  8. CF1060E Sergey and Subways 假的点分治

    题目传送门:http://codeforces.com/problemset/problem/1060/D 题意:给出$N$个点的一棵树,现在将距离为$2$的点之间连一条边,求所有点对之间最短路的和, ...

  9. prop和attr的比较

    prop来获取或设置固有属性  removeProp()  删除固有属性 attr来获取或设置自定义属性     removeAttr() 删除自定义属性 案例:全选与全不选 <body> ...

  10. 数列分块入门九题(二):LOJ6280~6282

    Preface 个人感觉这中间的三题是最水的没有之一 数列分块入门 4--区间加法,区间求和 这个也是很多数据结构完爆的题目线段树入门题,但是练分块我们就要写吗 修改还是与之前类似,只不过我们要维护每 ...