建议直接空降至参考文献,点击链接

简单工厂模式

#include<iostream>
using namespace std;
class BasicCamera {
public:
virtual ~BasicCamera() {};
virtual void OpenCamera() = 0;
}; class Hik :public BasicCamera {
public:
void OpenCamera() {
cout << "打开海康相机" << endl;
}
}; class DaHua :public BasicCamera {
public:
void OpenCamera() {
cout << "打开大华相机" << endl;
}
}; class Factory {
public:
BasicCamera* CreateCamera(string type) {
if (type == "Hik")
return new Hik();
else if (type == "DaHua")
return new DaHua();
else
return NULL; } };
int main() {
Factory fac = Factory();
BasicCamera* camera = fac.CreateCamera("Hik");
camera->OpenCamera();
delete camera;
getchar();
return 0;
}
当我们需要增加一类相机时,我们需要先定义一个类继承BasicCamera类,还要修改 Factory 类的代码,增加if else判断。这显然是违背开闭原则的。

Tips:

1. 由于工厂模式会返回一个基类指针,指向新生成的derived类对象,对象位于heap,为避免泄露内存和其他资源,要将返回的对象delete掉。如果前面的基类(BasicCamera)没有virtual的析构函数,那么通常发生的是对象的derived部分没有被销毁,产生局部销毁的现象。因此必须要加

virtual ~BasicCamera() {};

2. 基类生成对象不合理,因此在基类中定义了纯虚函数

virtual void OpenCamera() = 0;

3. UML图

+表示public,函数后面加冒号后面跟着返回类型。

工厂方法模式

#include<iostream>
using namespace std; class BasicCamera {
public:
virtual ~BasicCamera() {};
virtual void OpenCamera() = 0;
};
class Hik :public BasicCamera {
public:
void OpenCamera() {
cout << "打开海康相机" << endl;
}
};
class DaHua :public BasicCamera {
public:
void OpenCamera() {
cout << "打开大华相机" << endl;
}
}; class Factory {
public:
virtual ~Factory() {}
virtual BasicCamera* CreateCamera() = 0;
};
class HikFactory :public Factory {
BasicCamera* CreateCamera() { return new Hik(); }
};
class DaHuaFactory :public Factory {
DaHua* CreateCamera() { return new DaHua(); }
}; int main() {
Factory* fac = new HikFactory();
BasicCamera* cam = fac->CreateCamera();
cam->OpenCamera();
delete cam;
delete fac;
getchar();
return 0;
}

以下情况下可以使用工厂方法模式:

  • 一个类不知道它所需要的对象的类:在工厂方法模式中,客户端不需要知道具体产品类的类名,只需要知道所对应的工厂即可,具体的产品对象由具体工厂类创建;客户端需要知道创建具体产品的工厂类。
  • 一个类通过其子类来指定创建哪个对象:在工厂方法模式中,对于抽象工厂类只需要提供一个创建产品的接口,而由其子类来确定具体要创建的对象,利用面向对象的多态性和里氏代换原则,在程序运行时,子类对象将覆盖父类对象,从而使得系统更容易扩展。
  • 将创建对象的任务委托给多个工厂子类中的某一个,客户端在使用时可以无须关心是哪一个工厂子类创建产品子类,需要时再动态指定,可将具体工厂类的类名存储在配置文件或数据库中。

优点:

  • 一个调用者想创建一个对象,只要知道其名称就可以了。
  • 扩展性高,如果想增加一个产品,只要扩展一个工厂类就可以。
  • 屏蔽产品的具体实现,调用者只关心产品的接口。

缺点:

每次增加一个产品时,都需要增加一个具体类和对象实现工厂,使得系统中类的个数成倍增加,在一定程度上增加了系统的复杂度,同时也增加了系统具体类的依赖。这并不是什么好事。

抽象工厂模式

#include<iostream>
using namespace std;
class BasicCamera {
public:
virtual ~BasicCamera() {};
virtual void OpenCamera() = 0;
};
class HikCamera :public BasicCamera {
public:
void OpenCamera() {
cout << "打开海康相机" << endl;
}
}; class BasicLen {
public:
virtual ~BasicLen() {};
virtual void InstallLen() = 0;
};
class HikLen :public BasicLen {
void InstallLen() {
cout << "安装了海康镜头" << endl;
}
}; class Factory { public:
virtual ~Factory() {};
virtual BasicCamera* CreateCamera() = 0;
virtual BasicLen* CreateLen() = 0;
};
class HikFactory :public Factory {
BasicCamera* CreateCamera() { return new HikCamera(); }
BasicLen* CreateLen() { return new HikLen(); }
}; int main() {
Factory* fac = new HikFactory();
BasicCamera* cam = fac->CreateCamera();
BasicLen* len = fac->CreateLen();
cam->OpenCamera();
len->InstallLen();
delete len;
delete cam;
delete fac;
getchar();
return 0;
}

优点:

  • 抽象工厂模式隔离了具体类的生成,使得客户并不需要知道什么被创建。由于这种隔离,更换一个具体工厂就变得相对容易,所有的具体工厂都实现了抽象工厂中定义的那些公共接口,因此只需改变具体工厂的实例,就可以在某种程度上改变整个软件系统的行为。
  • 当一个产品族中的多个对象被设计成一起工作时,它能够保证客户端始终只使用同一个产品族中的对象。
  • 增加新的产品族很方便,无须修改已有系统,符合“开闭原则”。

缺点:

增加新的产品等级结构麻烦,需要对原有系统进行较大的修改,甚至需要修改抽象层代码,这显然会带来较大的不便,违背了“开闭原则”。

抽象模板工厂

#include<iostream>
using namespace std;
class BasicCamera {
public:
virtual ~BasicCamera() {};
virtual void OpenCamera() = 0;
};
class HikCamera :public BasicCamera {
public:
void OpenCamera() {
cout << "打开海康相机" << endl;
}
};
class DaHuaCamera :public BasicCamera {
public:
void OpenCamera() {
cout << "打开大华相机" << endl;
}
}; class BasicLen {
public:
virtual ~BasicLen() {};
virtual void InstallLen() = 0;
};
class HikLen :public BasicLen {
void InstallLen() {
cout << "安装了海康镜头" << endl;
}
}; template <class AbstractProduct_t>
class AbstractFactory {
public:
virtual ~AbstractFactory() {};
virtual AbstractProduct_t* CreateProduct() = 0;
};
template <class AbstractProduct_t, class ConcreteProduct_t>
class ConcreteFactory :public AbstractFactory< AbstractProduct_t> {
public:
AbstractProduct_t* CreateProduct() { return new ConcreteProduct_t(); }
}; int main() {
AbstractFactory<BasicCamera>* fac = new ConcreteFactory<BasicCamera, HikCamera>();
AbstractFactory<BasicCamera>* fac1 = new ConcreteFactory<BasicCamera, DaHuaCamera>();
AbstractFactory<BasicLen>* fac2 = new ConcreteFactory<BasicLen, HikLen>();
BasicCamera* cam = fac->CreateProduct();
BasicCamera* cam1 = fac1->CreateProduct();
BasicLen* len = fac2->CreateProduct();
cam->OpenCamera();
cam1->OpenCamera();
len->InstallLen();
delete cam,cam1,len,fac,fac1,fac2;
getchar();
return 0;
}

产品注册模板类+单例工厂模板类

#include<iostream>
#include <map>
using namespace std;
class BasicCamera {
public:
virtual ~BasicCamera() {};
virtual void OpenCamera() = 0;
};
class HikCamera :public BasicCamera {
public:
void OpenCamera() {
cout << "打开海康相机" << endl;
}
};
class DaHuaCamera :public BasicCamera {
public:
void OpenCamera() {
cout << "打开大华相机" << endl;
}
}; class BasicLen {
public:
virtual ~BasicLen() {};
virtual void InstallLen() = 0;
};
class HikLen :public BasicLen {
void InstallLen() {
cout << "安装了海康镜头" << endl;
}
}; // 基类,产品注册模板接口类
// 模板参数 ProductType_t 表示的类是产品抽象类
template <class ProductType_t>
class IProductRegistrar
{
public:
// 获取产品对象抽象接口
virtual ProductType_t* CreateProduct() = 0; protected:
// 禁止外部构造和虚构, 子类的"内部"的其他函数可以调用
IProductRegistrar() {}
virtual ~IProductRegistrar() {} private:
// 禁止外部拷贝和赋值操作
IProductRegistrar(const IProductRegistrar&);
const IProductRegistrar& operator=(const IProductRegistrar&);
}; // 工厂模板类,用于获取和注册产品对象
// 模板参数 ProductType_t 表示的类是产品抽象类
template <class ProductType_t>
class ProductFactory
{
public:
// 获取工厂单例,工厂的实例是唯一的
static ProductFactory<ProductType_t>& Instance()
{
static ProductFactory<ProductType_t> instance;
return instance;
} // 产品注册
void RegisterProduct(IProductRegistrar<ProductType_t>* registrar, std::string name)
{
m_ProductRegistry[name] = registrar;
} // 根据名字name,获取对应具体的产品对象
ProductType_t* GetProduct(std::string name)
{
// 从map找到已经注册过的产品,并返回产品对象
if (m_ProductRegistry.find(name) != m_ProductRegistry.end())
{
return m_ProductRegistry[name]->CreateProduct();
} // 未注册的产品,则报错未找到
std::cout << "No product found for " << name << std::endl; return NULL;
} private:
// 禁止外部构造和虚构
ProductFactory() {}
~ProductFactory() {} // 禁止外部拷贝和赋值操作
ProductFactory(const ProductFactory&);
const ProductFactory& operator=(const ProductFactory&); // 保存注册过的产品,key:产品名字 , value:产品类型
map<string, IProductRegistrar<ProductType_t>*> m_ProductRegistry;
};
// 产品注册模板类,用于创建具体产品和从工厂里注册产品
// 模板参数 ProductType_t 表示的类是产品抽象类(基类),ProductImpl_t 表示的类是具体产品(产品种类的子类)
template <class ProductType_t, class ProductImpl_t>
class ProductRegistrar : public IProductRegistrar<ProductType_t>
{
public:
// 构造函数,用于注册产品到工厂,只能显示调用
explicit ProductRegistrar(std::string name)
{
// 通过工厂单例把产品注册到工厂
ProductFactory<ProductType_t>::Instance().RegisterProduct(this, name);
} // 创建具体产品对象指针
ProductType_t* CreateProduct()
{
return new ProductImpl_t();
}
}; int main() {
IProductRegistrar<BasicCamera>* reg = new ProductRegistrar<BasicCamera, HikCamera>("camera_Hik");
BasicCamera* cam = ProductFactory<BasicCamera>::Instance().GetProduct("camera_Hik");
cam->OpenCamera();
if (cam) delete cam;
getchar();
return 0;
}

产品注册的对象用std::map的方式保存,通过key-valve的方式可以轻松简单的获取对应的产品对象实例。

Tips:(未完成)

1. Instance

单例模式

2. map

map是STL的一个关联容器,它提供一对一的hash

  • 第一个可以称为关键字(key),每个关键字只能在map中出现一次;
  • 第二个可能称为该关键字的值(value);

参考文献

https://www.cnblogs.com/xiaolincoding/p/11524376.html

https://www.cnblogs.com/xiaolincoding/p/11524401.html

https://www.zhihu.com/question/20367734/answer/1089721250

C++笔记(11)工厂模式的更多相关文章

  1. Java学习笔记——Java工厂模式之简单工厂

    package com.app; import java.util.Date; /* * 工厂模式:简单工厂.工厂方法.抽象工厂 * * */ public class Test0718_Factor ...

  2. HeadFirst设计模式读书笔记(4)-工厂模式

    工厂方法模式:定义了一个创建对象的接口,但由子类决定要实例化的类是哪一个.工厂方法让类把实例化推迟到子类. 所有工厂模式都用来封装对象的创建.工厂方法模式通过让子类决定该创建的对象是什么,来达到将对象 ...

  3. 学习笔记——抽象工厂模式Abstract Factory

    在工厂模式的基础上,通过为工厂类增加接口,实现其他产品的生产,而不用一类产品就增加一个工厂. 依然以<真菌世界>游戏故事类比,树作为工厂,如果现在有两类树,一类生产快速弄真菌飞机和20毫米 ...

  4. C#学习笔记-抽象工厂模式

    题目1:数据访问,通过数据库对用户表单的进行访问,数据库包含SQL Server,对用户表单进行“新增用户”和“查询用户”信息等操作. 分析: 首先,确认用户表单,里面包含两个ID和Name两个字段, ...

  5. 设计模式之笔记--抽象工厂模式(Abstract Factory)

    抽象工厂模式(Abstract Factory) 定义 抽象工厂模式(Abstract Factory),提供一个创建一系列相关或相互依赖对象的接口,而无需指定它们具体的类. 类图 描述 多个抽象产品 ...

  6. 设计模式之笔记--简单工厂模式(Simple Factory)

    简单工厂模式(Simple Factory) 类图 描述 简单工厂: 一个抽象产品类,可以派生多个具体产品类: 一个具体工厂类: 工厂只能创建一个具体产品. 应用场景 汽车接口 public inte ...

  7. 设计模式 笔记 抽象工厂模式 Abstract Factory

    //---------------------------15/04/09---------------------------- //Abstract Factory 抽象工厂----对象创建型模式 ...

  8. HeadFirst设计模式读书笔记之工厂模式

    1. 简单工厂 1. 你开了一家披萨店,点披萨的方法可能是这样: public Pizza orderPizza(String type) { Pizza pizza; if (type.equals ...

  9. 《图解设计模式》读书笔记1-1 Iterator模式

    目录 迭代器模式的类图 类图的解释 迭代器模式的代码 解释 原因 思想 迭代器模式的类图 类图的解释 名称 说明 Aggregate 集合接口,有提供迭代器的方法 Iterator 迭代器接口,提供迭 ...

  10. 工厂模式 - Factory

    简单工厂模式 SimpleFactory Pattern,将一个具体类的实例化交给一个静态工厂方法来执行. 特点: 增加功能需要修改工厂类,扩展性较差: 参考: 设计模式学习笔记 - 简单工厂模式: ...

随机推荐

  1. 重新整理数据结构与算法(c#)——算法套路迪杰斯特拉算法[三十一]

    前言 迪杰斯特拉算法 是求最短路径方法的其中一种,这个有什么作用呢? 有一张图: 假设求G点到其他各点的最小路径. 是这样来的. 比如找到了和G点相连接所有点,ABED.这时候确定GA是一定是最短的, ...

  2. java操作xml超简单的方法

    用dom4j?SAX?no,no,no,光看api和帮助文档就烦,有没有更简单的方法呢?答案是有的. 那就是默默无名的:JAXB jaxb是啥? 摘抄一段度娘百科的介绍: JAXB能够使用Jackso ...

  3. Oracle SQL 创建一个简单的存储过程procedure

    Oracle 简单的创建一个存储过程procedure 如果学过别的语言,例如java,c这些,那么其实很好理解,其实就是面向数据库的操作 简单的例子如下: --创建或者重写存储过程 create o ...

  4. js 连接数据库 提示:ActiveXObject is not defined

    ActiveXObject is not defined 最近比较闲,上班瞎捣鼓一下,没想到报错了,提示ActiveXObject is not defined 大概是在js连接数据库时new对象使用 ...

  5. 单元测试必备:Asp.Net Core代码覆盖率实战,打造可靠应用 !

    引言 在前几章我们深度讲解了单元测试和集成测试的基础知识,这一章我们来讲解一下代码覆盖率,代码覆盖率是单元测试运行的度量值,覆盖率通常以百分比表示,用于衡量代码被测试覆盖的程度,帮助开发人员评估测试用 ...

  6. 力扣1132(MySQL)-报告的记录Ⅱ(中等)

    题目: 编写一段 SQL 来查找:在被报告为垃圾广告的帖子中,被移除的帖子的每日平均占比,四舍五入到小数点后 2 位. Actions 表: Removals 表: Result 表: 2019-07 ...

  7. 第 8章 Python 爬虫框架 Scrapy(下)

    第 8章 Python 爬虫框架 Scrapy(下) 8.1 Scrapy 对接 Selenium 有一种反爬虫策略就是通过 JS 动态加载数据,应对这种策略的两种方法如下:  分析 Ajax 请求 ...

  8. 同程旅行基于 RocketMQ 高可用架构实践

    ​简介: 我们在几年前决定引入 MQ 时,市场上已经有不少成熟的解决方案,比如 RabbitMQ , ActiveMQ,NSQ,Kafka 等.考虑到稳定性.维护成本.公司技术栈等因素,我们选择了 R ...

  9. Joint Consensus两阶段成员变更的单步实现

    ​简介: Raft提出的两阶段成员变更Joint Consensus是业界主流的成员变更方法,极大的推动了成员变更的工程应用.但Joint Consensus成员变更采用两阶段,一次变更需要提议两条日 ...

  10. Etcd 可视化管理工具,GUI 客户端。

    Etcd Assistant--Etcd 可视化管理工具,GUI 客户端. 下载地址:http://www.redisant.cn/etcd 主要功能: 支持多标签页,同时连接到多个集群 以漂亮的格式 ...