C++去掉宏的自注册工厂

2024-08-31 11:12
文章标签 c++ 注册 工厂 去掉

本文主要是介绍C++去掉宏的自注册工厂,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

实现思路

借助CRTP和静态变量的初始化顺序来实现

静态变量的初始化是在main函数之前,利用静态变量的初始化实现自注册

实现代码

#include <memory>
#include <map>
#include <functional>
#include <iostream>struct message_base;template<typename... Args>
class MsgFactory {
public:MsgFactory() = delete;using CreateMethod = std::function<std::unique_ptr<message_base>(std::tuple<Args...>)>;static bool Register(size_t id, CreateMethod creator){auto& map = GetMap();auto it = map.find(id);if (it == map.end()){map[id] = creator;std::cout << id << " registered\n";return true;}return false;}static std::unique_ptr<message_base> Create(size_t id, Args... args){auto& map = GetMap();auto it = map.find(id);if (it != map.end())return it->second(std::make_tuple(args...));return nullptr;}static std::map<size_t, CreateMethod>& GetMap() {static std::map<size_t, CreateMethod> map;return map;}
};template <typename T, typename... Args>
class AutoMsgFactory
{
protected:static bool registered_;
};template<typename T, typename... Args>
std::unique_ptr<T> create(std::tuple<Args...> t) {return std::apply([](Args... args) {return std::make_unique<T>(args...);}, t);
}template <typename T, typename... Args>
bool AutoMsgFactory<T, Args...>::registered_ = MsgFactory<Args...>::Register(T::Id(), [](std::tuple<Args...> args) { return create<T>(args); });struct message_base {virtual ~message_base() {}
};struct message1 : public message_base, AutoMsgFactory<message1>
{static size_t Id() {assert(!registered_);//registered_保证当前类会被自动注册//避免重复注册,重复注册时会触发断言错误//编译期限定派生类必须定义一个Id函数,保证派生类不会忘记定义创建该类需要的唯一id//id是自定义的,可以是枚举类型也可以是整形return 1;}
};struct message2 : public message_base, AutoMsgFactory<message2>
{static size_t Id() {assert(!registered_);return 2;}
};struct message4 : public message_base, AutoMsgFactory<message4, int>
{message4(int) {}static size_t Id() {assert(registered_);return 4;}
};auto msg = MsgFactory<int>::Create(4, 1);
if (msg) {std::cout << typeid(msg).name() << std::endl;
}auto msg1 = MsgFactory::Create(1);
if (msg1) {std::cout << typeid(msg1).name() << std::endl;
}auto msg2 = MsgFactory::Create(2);
if (msg2) {std::cout << typeid(msg2).name() << std::endl;
}

这篇关于C++去掉宏的自注册工厂的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1123740

相关文章

使用C++实现链表元素的反转

《使用C++实现链表元素的反转》反转链表是链表操作中一个经典的问题,也是面试中常见的考题,本文将从思路到实现一步步地讲解如何实现链表的反转,帮助初学者理解这一操作,我们将使用C++代码演示具体实现,同... 目录问题定义思路分析代码实现带头节点的链表代码讲解其他实现方式时间和空间复杂度分析总结问题定义给定

C++初始化数组的几种常见方法(简单易懂)

《C++初始化数组的几种常见方法(简单易懂)》本文介绍了C++中数组的初始化方法,包括一维数组和二维数组的初始化,以及用new动态初始化数组,在C++11及以上版本中,还提供了使用std::array... 目录1、初始化一维数组1.1、使用列表初始化(推荐方式)1.2、初始化部分列表1.3、使用std::

C++ Primer 多维数组的使用

《C++Primer多维数组的使用》本文主要介绍了多维数组在C++语言中的定义、初始化、下标引用以及使用范围for语句处理多维数组的方法,具有一定的参考价值,感兴趣的可以了解一下... 目录多维数组多维数组的初始化多维数组的下标引用使用范围for语句处理多维数组指针和多维数组多维数组严格来说,C++语言没

Go路由注册方法详解

《Go路由注册方法详解》Go语言中,http.NewServeMux()和http.HandleFunc()是两种不同的路由注册方式,前者创建独立的ServeMux实例,适合模块化和分层路由,灵活性高... 目录Go路由注册方法1. 路由注册的方式2. 路由器的独立性3. 灵活性4. 启动服务器的方式5.

c++中std::placeholders的使用方法

《c++中std::placeholders的使用方法》std::placeholders是C++标准库中的一个工具,用于在函数对象绑定时创建占位符,本文就来详细的介绍一下,具有一定的参考价值,感兴... 目录1. 基本概念2. 使用场景3. 示例示例 1:部分参数绑定示例 2:参数重排序4. 注意事项5.

使用C++将处理后的信号保存为PNG和TIFF格式

《使用C++将处理后的信号保存为PNG和TIFF格式》在信号处理领域,我们常常需要将处理结果以图像的形式保存下来,方便后续分析和展示,C++提供了多种库来处理图像数据,本文将介绍如何使用stb_ima... 目录1. PNG格式保存使用stb_imagephp_write库1.1 安装和包含库1.2 代码解

C++实现封装的顺序表的操作与实践

《C++实现封装的顺序表的操作与实践》在程序设计中,顺序表是一种常见的线性数据结构,通常用于存储具有固定顺序的元素,与链表不同,顺序表中的元素是连续存储的,因此访问速度较快,但插入和删除操作的效率可能... 目录一、顺序表的基本概念二、顺序表类的设计1. 顺序表类的成员变量2. 构造函数和析构函数三、顺序表

使用C++实现单链表的操作与实践

《使用C++实现单链表的操作与实践》在程序设计中,链表是一种常见的数据结构,特别是在动态数据管理、频繁插入和删除元素的场景中,链表相比于数组,具有更高的灵活性和高效性,尤其是在需要频繁修改数据结构的应... 目录一、单链表的基本概念二、单链表类的设计1. 节点的定义2. 链表的类定义三、单链表的操作实现四、

使用C/C++调用libcurl调试消息的方式

《使用C/C++调用libcurl调试消息的方式》在使用C/C++调用libcurl进行HTTP请求时,有时我们需要查看请求的/应答消息的内容(包括请求头和请求体)以方便调试,libcurl提供了多种... 目录1. libcurl 调试工具简介2. 输出请求消息使用 CURLOPT_VERBOSE使用 C

C++实现获取本机MAC地址与IP地址

《C++实现获取本机MAC地址与IP地址》这篇文章主要为大家详细介绍了C++实现获取本机MAC地址与IP地址的两种方式,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 实际工作中,项目上常常需要获取本机的IP地址和MAC地址,在此使用两种方案获取1.MFC中获取IP和MAC地址获取