如何实现自动注册对象Factory

目标

了解C++里的Factory模式应用场景、实现方法,采用模板实现带来的便利。

来源

用途

一系列类继承自基类,均可以执行某个动作,可以根据类的Key获取类实例进行动作。

原始版本及存在的问题

class IObject
{
public:
    virtual void run() = 0;
    virtual ~IObject() {};
};
 
class Object1:public IObject
{
public:
    ~Object1()
    {
        ;
    }
    virtual void run() override
    {
        std::cout<<"Object1\n";
    }
};
 
class Object2 :public IObject
{
public:
    ~Object2()
    {
        ;
    }
 
    virtual void run() override
    {
        std::cout << "Object2\n";
    }
};
 
class ObjectFactory
{
public:
    static std::shared_ptr<IObject> create(const std::string& key)
    {
        if(key == "Object1")
            return std::make_shared<Object1>();
        if(key == "Object2")
            return std::make_shared<Object2>();
        return nullptr;
    }
};
 
void original_test()
{
    auto pVal1 = ObjectFactory::create("Object1");
    auto pVal2 = ObjectFactory::create("Object2");
 
    if(pVal1)
        pVal1->run();
    if(pVal2)
        pVal2->run();
}

这是最常规的实现方法,一旦新增类,则需要调整factory的创建方法,一旦这些代码被封装成库就没有扩展的可能;
我们希望没有非必要的依赖,避免调整类定义,尽可能简单就能够注册新类。

实现思路

  1. 准备基类
  2. Factory单例
  3. 注册类构造方法
  4. 获取类实例

调整版本

基类

class IObject
{
public:
    virtual void run() = 0;
    virtual ~IObject() {};
};

Factory单例

class ObjectFactory
{
public:
    static ObjectFactory* Instance()
    {
        static ObjectFactory factory;
        return &factory;
    }
};

注册类构造方法

    void registerObjectFunc(const std::string& name, std::function<IObject*()> func)
    {
        funcs_[name] =func;
    }
private:
    std::map<std::string,std::function<IObject*()>> funcs_;

构造类实例

std::shared_ptr<IObject> create(const std::string& key)
{
    auto it = funcs_.find(key);
    if (it != funcs_.end())
        return std::shared_ptr<IObject>(it->second());
    return nullptr;
}

使用方法

//类1
class Object1 :public IObject
{
public:
    ~Object1()
    {
        ;
    }
    virtual void run() override
    {
        std::cout << "Object1\n";
    }
};
//类2
class Object2 :public IObject
{
public:
    ~Object2()
    {
        ;
    }
 
    virtual void run() override
    {
        std::cout << "Object2\n";
    }
};

//使用
auto pVal1= ObjectFactory::Instance()->create("Object1");
auto pVal2 = ObjectFactory::Instance()->create("Object2");
 
if (pVal1)
    pVal1->run();
if (pVal2)
    pVal2->run();

新类注册

static ObjectRegisterHelper Object1_ObjectRegisterHelper("Object1", []()->IObject* { return new Object1();  });
static ObjectRegisterHelper Object2_ObjectRegisterHelper("Object2", []()->IObject* { return new Object2();  });

使用宏进行新类注册

辅助注册:

class ObjectRegisterHelper
{
public:
    ObjectRegisterHelper(const char* key, std::function<IObject*()> func)
    {
        ObjectFactory::Instance()->registerObjectFunc(key,func);
    }
};
 
#define REGISTER_OBJECT(className,key) \
static ObjectRegisterHelper className##ObjectRegisterHelper(key,[]()->className*{ return new className();})

使用方式:

REGISTER_OBJECT(Object1,"Object1");
REGISTER_OBJECT(Object2, "Object2");

使用模板进行新类注册

辅助注册:

template<typename T>
class ObjectRegister
{
public:
    ObjectRegister(const char* key)
    {
        ObjectFactory::Instance()->registerObjectFunc(key,[](){
            return new T();
        });
    }
};

使用方式:

static ObjectRegister<Object1> register1("Object1");
static ObjectRegister<Object2> register2("Object2");

自动注册的实现

以上的方案都需要在外部定义静态变量用来进行构造方法注册,还有方法无需外部定义:

将静态变量包裹在类中

约束Key获取接口

每个类定义均有静态接口Key()来获取Key供注册使用

template<typename T>
void registerT()
{
    //std::cout << "RegisterT " << T::Key() << "\n";
    funcs_[T::Key()] = []()->IEntry* { return new T(); };
}

定义辅助类

template<typename T>
struct RegisterClass
{
    RegisterClass()
    {
        Factory::Instance().registerT<T>();
    }
};

自动注册类

I为基类,新类继承自AutoRegister自动附带注册类

template<typename T, typename I>
struct AutoRegister :public I
{
    AutoRegister()
    {
        //®ister_;
    }
public:
    static RegisterClass<T> register_;
};

//模板类静态变量声明->(C++14的变量模板)
template<typename T, typename I>
RegisterClass<T> AutoRegister<T, I>::register_;

使用

class Object :public AutoRegister<Object, IEntry>
{
public:
    void run()
    {
        std::cout << "Object\n";
    }
    //Key配置接口
    static const char* Key()
    {
        //由于VisualStudio2015不支持变量模板,这个可以用来声明静态变量
        Object::register_;
        //在支持变量模板的C++编译器中这个是不需要的
        return "Object";
    }
};

调整注册辅助类

C++11新特性:内部类可以通过外部类的实例访问外部类的私有成员;

struct Factory
{
public:
    template<typename T>
    struct register_helper
    {
        register_helper(const char* key)
        {
            //获取外部类实例的私有成员变量
            Instance().funcs_[key] = []()->IEntry* { return new T(); };
        }
    };

模板注册免命名

之前使用宏进行新类注册会自动合并出静态辅助类实例名称,而使用模板进行新类注册则需要保证名称不重复,否则编译报错,可以采用以下方式避免该问题:

template<typename T>
struct register_agent
{
    static Factory::register_helper<T> helper;
};

//TObject的注册辅助类静态实例
decltype(register_agent<TObject>::helper) register_agent<TObject>::helper("TObject");
//Object的
decltype(register_agent<Object>::helper) register_agent<Object>::helper("Object");

其它调整

参见C++11实现一个自动注册的工厂

通过实现factory模板能够实现统一的factory。

学到的内容

  • Factory模式的实现
  • 静态变量在自动注册等处理方面的应用
  • 在一些场景下如何使用模板替代宏
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,732评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 87,496评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,264评论 0 338
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,807评论 1 277
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,806评论 5 368
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,675评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,029评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,683评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 41,704评论 1 299
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,666评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,773评论 1 332
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,413评论 4 321
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,016评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,978评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,204评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,083评论 2 350
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,503评论 2 343

推荐阅读更多精彩内容

  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,596评论 18 139
  • 1. Java基础部分 基础部分的顺序:基本语法,类相关的语法,内部类的语法,继承相关的语法,异常的语法,线程的语...
    子非鱼_t_阅读 31,573评论 18 399
  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 171,448评论 25 707
  • Spring Boot 参考指南 介绍 转载自:https://www.gitbook.com/book/qbgb...
    毛宇鹏阅读 46,733评论 6 342
  • 对应的水题是poj3468 今天实验室的大牛说了线段树的区间修改值在求和,(其实自己线段树还没懂太多了)觉得他们好...
    Anxdada阅读 888评论 0 0