c++模板实现抽象工厂
类似于rime的rime::Class<factory type, product type>实现方式。
C++模板实现的通用工厂方法模式
1.工厂方法(Factory Method)模式
工厂方法模式的意义是定义一个创建产品对象的工厂接口,将实际创建工作推迟到子类当中。核心工厂类不再负责产品的创建,这样核心类成为一个抽象工厂角色,仅负责具体工厂子类必须实现的接口,这样进一步抽象化的好处是使得工厂方法模式可以使系统在不修改具体工厂角色的情况下引进新的产品。

工厂方法模式结构示意图
工厂方法模式是简单工厂模式的衍生,解决了许多简单工厂模式的问题。首先完全实现‘开-闭 原则’,实现了可扩展。其次更复杂的层次结构,可以应用于产品结果复杂的场合。
工厂方法模式很好用,但在代码层面上,为每一个产品都编写一个对应的工厂,这无疑很麻烦,而且不同的Product要写不同Creator。想想你在写一个GUI库,基本控件为Widget类,由其派生下了TextBox,Button,Label等几十个控件,那你需要为这几十个控件分别编写对应的ConcreteCreator。天,这太麻烦了吧!而如果你还要写一个RPG小游戏,需要从Person类派生下如NPC,Hero,Emeny,Ally等又是十几个类,而且还要编写一个对应于Person的Creator....
2.模板实现工厂方法模式
面对这种类型造成的多态,C++模板就要出场了。现在情况是这样的,Product和ConcreteProduct都已经写好,我们要为其编写对应的工厂类。可以看出来,Product对应对Creator,ConcreteProduct对应于ConcreteCreator。也就是说,如果想编写通用的ConcreteCreator,就得将ConcreteProduct抽象,想编写通用的Creator,就得将Product抽象。而我们最后会将这2者都抽象了。
首先编写通用的ConcreteCreator
- // 具体工厂类
- // Product为抽象产品,ConcreteProduct为具体产品
- template<typename Product, typename ConcreteProduct>
- class ConcreteCreator
- {
- public:
- // 生产一个具体产品
- static Product* createProduct()
- {
- return new ConcreteProduct();
- }
- };
注意createProduct()是静态的,是因为该函数要被保存起来的,静态函数只需保存其函数指针即可,而普通的成员函数还要额外保存一个对象指针,这无疑更麻烦了。 也就是我们不需要这样用:
- ConcreteCreator<Product, ConcreteProduct> concreteCreator;
- Product* p = concreteCreator.createProduct();
只需要这样写:
- Product* p = ConcreteCreator<Product, ConcreteProduct>::createProduct();
下面是Creator的实现
- // 抽象工厂
- // Product为抽象产品
- template<typename Product>
- class Creator
- {
- // 单例实现
- public:
- static Creator& Instance()
- {
- static Creator<Product> instance;
- return instance;
- }
- private:
- Creator() {}
- ~Creator() {}
- Creator(Creator&);
- // 对外接口
- public:
- typedef Product* (*CreateProductDelegate)( ); // 生产产品的产品
- typedef std::map<std::string, CreateProductDelegate> MapRegisterCreatorItem;
- // 根据具体产品生成具体工厂
- // 并将具体产品注册进抽象工厂
- // ConcreteProduct为具体产品
- template<typename ConcreteProduct>
- void registerCreator(const std::string& _type)
- {
- mConcreteCreators[_type] = ConcreteCreator<Product, ConcreteProduct>::createProduct;
- }
- // 删除所有具体工厂
- void unregisterAllCreators()
- {
- mConcreteCreators.clear();
- }
- // 生产类型为_type的产品
- // 失败返回0
- Product* createProduct(const std::string& _type)
- {
- MapRegisterCreatorItem::iterator type = mConcreteCreators.find(_type);
- if (type != mConcreteCreators.end())
- {
- CreateProductDelegate create = type->second;
- if (create != 0)
- return create();
- }
- return 0;
- }
- private:
- MapRegisterCreatorItem mConcreteCreators; // 保存所有注册过的具体产品
- };
下面来简单解释一下上面的代码。
首先Creator实现了单例模式,这只是为了方便使用,你也可以不实现为单例。
Creator里面保存了所有注册过的具体工厂,具体工厂在注册时被构建,每个具体工厂对应一个名字(string类型)。
createProduct即为工厂方法,外部通过此接口创建产品,创建时仅需提供具体工厂的名字即可。
我们以RPG游戏那个作为例子。要创建一个Person的Creator,可以这样写
- typedef Creator<Person> PersonCreator;
- PersonCreator& factory = PersonCreator::Instance();
- factory.registerCreator<Person>("Person");
- factory.registerCreator<Hero>("Hero");
- factory.registerCreator<NPC>("NPC");
这样即完成了注册过程,你可以继续为其他类型注册具体工厂。
要创建一个NPC,可以这样写
- Person* npc = factory.createProduct("NPC");
3.优缺点
此方法的优点很明显,就是用起来很方便。由于Creator支持不同类型,你不要为不同的产品编写不同的Creator。而且其提供了注册具体工厂的方法,你仅需要写一行代码即可为具体产品生成具体的工厂,而不需要编写专门的类。
而且其注册过程是可以在运行时修改的,也就是说你可以在运行时动态地注册或反注册具体工厂,这样灵活性就很大了。
缺点的话就是不支持构造函数参数,也就是说在创建产品时不支持参数。当然可以为不同数量参数编写具体的模板Creator。但其实个人并不提倡提供无参构造函数。我认为像这类产品类,都应该提供一个无参的构造函数版本,然后提供get,set函数来修改内部成员变量。
c++模板实现抽象工厂的更多相关文章
- 设计模式之美:Abstract Factory(抽象工厂)
索引 别名 意图 结构 参与者 适用性 缺点 效果 相关模式 命名约定 实现 实现方式(一):使用 Factory Method 来实现 Abstract Factory. 实现方式(二):使用 Pr ...
- 5. 星际争霸之php设计模式--抽象工厂模式
题记==============================================================================本php设计模式专辑来源于博客(jymo ...
- 利用CodeSmith生成抽象工厂步骤
其实CodeSmith挺好的,帮我们主动生成不少代码,并且代码质量不错,下面就来介绍一下利用CodeSmith生成抽象工厂步骤 打开codesmith模板的buildall 注意path的设置,因为后 ...
- Java设计模式系列-抽象工厂模式
原创文章,转载请标注出处:https://www.cnblogs.com/V1haoge/p/10755412.html 一.概述 抽象工厂模式是对工厂方法模式的再升级,但是二者面对的场景稍显差别. ...
- 第20月第29天 cocoa抽象工厂 cocoapods组件化 cocoapods升级
1. 在 Cocoa Touch 框架中,类簇是抽象工厂模式在 iOS 下的一种实现,以 NSArray 举例,将原有的 alloc+init 拆开写: id obj1 = [NSArray allo ...
- 设计模式C++学习笔记之七(AbstractFactory抽象工厂模式)
抽象工厂,提供一个创建一系列相关或相互依赖对象的接口,而无需指定它们具体的类.对于工厂方法来说,抽象工厂可实现一系列产品的生产,抽象工厂更注重产品的组合. 看代码: 7.1.解释 main(),女 ...
- 【设计模式】 模式PK:抽象工厂模式VS建造者模式
1.概述 抽象工厂模式实现对产品家族的创建,一个产品家族是这样的一系列产品:具有不同分类维度的产品组合,采用抽象工厂模式则是不需要关心构建过程,只关心什么产品由什么工厂生产即可.而建造者模式则是要求按 ...
- 设计模式C#合集--抽象工厂模式
抽象工厂,名字就告诉你是抽象的了.上代码. public interface BMW { public void Drive(); } public class BMW730 : BMW { publ ...
- PHP设计模式(三)抽象工厂模式(Abstract Factory For PHP)
一.什么是抽象工厂模式 抽象工厂模式的用意为:给客户端提供一个接口,可以创建多个产品族中的产品对象 ,而且使用抽象工厂模式还要满足以下条件: 系统中有多个产品族,而系统一次只可能消费其中一族产品. 同 ...
随机推荐
- Sophos UTM WebAdmin存在未明漏洞
...
- (三)JAVA使用POI操作excel
1,单元格对齐方式 Demo8.java package com.wishwzp.poi; import java.io.FileOutputStream; import java.util.Date ...
- 解决div布局中第一个div的margin-top在浏览器中显示无效果问题。
原味来源:http://www.hicss.net/do-not-tell-me-you-understand-margin/ 垂直外边距合并问题 别被上面这个名词给吓倒了,简单地说,外边距合并指的是 ...
- away3d打包到IOS锯齿问题解决办法
很多ios设备是高清屏,一个像素顶普通设备四个像素.从这点上也许可以入手解决. 先把<requestedDisplayResolution>high</requestedDispla ...
- global, $GLOBALS[]
// global在函数中产生一个指向函数外部变量的别名变量,而不是真正的函数外部变量,一旦改变了别名的变量指向地址,就会发生一些意外的情况 $a = 10; function test() { gl ...
- AVAudioRecorder 录制音频
AVFoundation 中使用AVAudioRecorder 类添加音频录制功能是非常简单的, AVAudioRecorder构建与Audio Queue Services之上是一个功能强大且代码简 ...
- Cassandra1.2文档学习(6)—— 客户端数据请求
参考文档:http://www.datastax.com/documentation/cassandra/1.2/webhelp/index.html#cassandra/architecture/a ...
- css实现div块半透明
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/ ...
- Android线程池的使用(未完)
ExecutorService Executors public class Executors // 创建一个线程池,使用固定数量的线程操作共享无界队列. public static Executo ...
- Open CASCADE 基础类(Foundation Classes)
1 介绍(Introduction) 1 如何使用Open CASCADE技术(OCCT)基础类. This manual explains how to use Open CASCADE Techn ...