c++工厂注册类

工厂注册类

利用模版形式注册类

cpp 复制代码
#include <iostream>
#include <memory>
#include <functional>
namespace cyn {


//自定义断言
//#ifndef _DEBUG // _RELEASE  或者 _DEBUG  ,根据你的编译器/构建系统
#ifdef _DEBUG // _RELEASE  或者 _DEBUG  ,根据你的编译器/构建系统
#define CHECK(cond) \
    do { \
    if (!(cond)) { \
    std::cerr << "Assertion failed: (" << #cond << "), function " << __FUNCTION__ \
    << ", file " << __FILE__ << ", line " << __LINE__ << "." << std::endl; \
} \
} while (0)
#define CHECK_EX(cond, msg) \
    do { \
    if (!(cond)) { \
    std::cerr << "Assertion failed: (" << #cond << "), function " << __FUNCTION__ \
    << ", file " << __FILE__ << ", line " << __LINE__ << "." << std::endl; \
    std::cerr << "Message: " << msg << std::endl; \
    throw std::runtime_error(msg); \
} \
} while (0)
#else
#define CHECK(cond) \
    do { \
    \
} while (0)
#define CHECK_EX(cond, msg) \
    do { \
    \
} while (0)
#endif



// 参数基类
class LayerParams
{
public:
    virtual ~LayerParams() = default;
};
// 具体参数类型
class DenseLayerParams : public LayerParams {
public:
    int units;
    float learning_rate;

    DenseLayerParams(int units, float learning_rate) : units(units), learning_rate(learning_rate) {}
};

// 基类
class Layer
{
public:
    Layer(const std::string& name,const std::shared_ptr<LayerParams>& params) : name(name),layer_params(params){}

    virtual int forward(int){return 0;};


public:
    int input_;
    std::string name;
    std::shared_ptr<LayerParams> layer_params;
};

//获取类型名字
template <typename T>
constexpr auto class_type_name() noexcept
{
    std::string_view name, prefix, suffix;
#ifdef _MSC_VER
    name = __FUNCSIG__;
    prefix = "auto __cdecl cyn::class_type_name<class cyn::";
    suffix = ">(void) noexcept";
#elif defined(__GNUC__)
    name = __PRETTY_FUNCTION__;
    prefix = "constexpr auto cyn::class_type_name() [with T = cyn::";
    suffix = "]";
#elif defined(__clang__)
    name = __PRETTY_FUNCTION__;
    prefix = "constexpr auto cyn::class_type_name() [with T = cyn::";
    suffix = "]";

#endif
    name.remove_prefix(prefix.size());
    name.remove_suffix(suffix.size());
    return name;
}

// 定义一个注册表基类模版
template <typename T, typename... Args>
class Factory
{
    using CreateFunction = std::function<std::shared_ptr<T>(Args...)>;
    Factory(){}

public:
    virtual ~Factory() = default;

    static Factory& Instance()
    {
        static Factory instance;
        return instance;
    }
    void Register(const std::string & name,const CreateFunction& create)
    {
        std::cout <<"Factory Register name:"<< std::string(name) << std::endl;
        registry[std::string(name)] = create;
    }

    std::shared_ptr<T> Create(const std::string& name,Args... args)
    {

        if (registry.find(name) != registry.end())
        {
            std::cout <<"Create succeed:"<< std::string(name) << std::endl;
            return registry[name](args...);
        }
        std::cout <<"Create fail:"<< std::string(name) << std::endl;
        return nullptr;
    }
private:
    std::unordered_map<std::string, CreateFunction> registry;
};
//注册类 统一接口
template <typename Base, typename Impl, typename... Args>
class Register
{
public:
    explicit Register()
    {
        std::cout<<"Register name:  "<<std::string(class_type_name<Impl>())<<std::endl;
        //流程6
        Factory<Base, Args...>::Instance().Register(std::string(class_type_name<Impl>()), [](Args... args)
        {
            return std::shared_ptr<Base>(new Impl(args...));
        });
    }
};











// LayerSub  在这很关键,实现了继承LayerSub  (也就是继承基类)自动注册
//注册Layer子类,实现继承自动注册  T 为Layer子类类型
/*
 * 继承LayerSub-构造函数->(void)registered->trigger()->_register(str)
 */
template <typename T>
class LayerSub : public Layer
{
    //流程4
    static bool trigger()
    {
        //流程5
        Register<Layer,T,std::shared_ptr<LayerParams>> _register;
        return true;
    }
protected:
    //流程2
    LayerSub(const std::shared_ptr<LayerParams>& params) : Layer(std::string(class_type_name<T>()),params)
    {
        std::cout<<"LayerSub:"<<std::string(class_type_name<T>())<<std::endl;

        //流程3
        (void)registered;
    }

public:
    static bool registered;

};
template <typename T>
bool LayerSub<T>::registered = LayerSub<T>::trigger();

//定义Layer类 创建接口
typedef Factory<Layer, std::shared_ptr<LayerParams>> LayerFactory;



// 具体层类型

class DenseLayer : public LayerSub<DenseLayer>
{
public:
    DenseLayer(const std::shared_ptr<LayerParams>& params) : LayerSub(params)
    {
        std::cout<<"DenseLayer constructor!"<<std::endl;
        std::shared_ptr<DenseLayerParams> p = std::dynamic_pointer_cast<DenseLayerParams>(params);

        //检查参数类型 - 调试用
        CHECK_EX(p,"error");


        //打印参数值
        std::cout<<p->units<<std::endl;
        std::cout<<p->learning_rate<<std::endl;
    }
    int forward(int s) override  {

        std::cout<<"wo shi zi lei!"<<s<<std::endl;
        return s;
    }
};

}

int main(int argc, char *argv[])
{
    QCoreApplication a(argc, argv);

    std::shared_ptr<cyn::DenseLayerParams> param = std::make_shared<cyn::DenseLayerParams>(100,1);
    auto ss = cyn::LayerFactory::Instance().Create(std::string("DenseLayer"), param);

    ss->forward(1);
    return a.exec();
}

输出结果:

Register name: DenseLayer

Factory Register name:DenseLayer

Create succeed:DenseLayer

LayerSub:DenseLayer

DenseLayer constructor!

100

1

wo shi zi lei!1

相关推荐
Smart-Space几秒前
cpphtmlbuilder-c++灵活构造html
c++·html
会叫的恐龙几秒前
C++ 核心知识点汇总(第四日)(循环结构)
c++·算法·循环结构
lly2024065 分钟前
HTML DOM 访问
开发语言
落羽的落羽5 分钟前
【Linux系统】文件IO:理解文件描述符、重定向、缓冲区
linux·服务器·开发语言·数据结构·c++·人工智能·机器学习
.小墨迹12 分钟前
apollo中速度规划的s-t图讲解【针对借道超车的问题】
开发语言·数据结构·c++·人工智能·学习
小龙报13 分钟前
【数据结构与算法】单链表的综合运用:1.合并两个有序链表 2.分割链表 3.环形链表的约瑟夫问题
c语言·开发语言·数据结构·c++·算法·leetcode·链表
拼好饭和她皆失14 分钟前
图论:最小生成树,二分图详细模板及讲解
c++·算法·图论
阿猿收手吧!15 分钟前
【C++】C++原子类型隐式转换解析
java·c++
爱喝水的鱼丶16 分钟前
SAP-ABAP:高效开发指南:全局唯一标识符ICF_CREATE_GUID函数的全面解析与实践
运维·服务器·开发语言·数据库·sap·abap·开发交流
HL_风神16 分钟前
C++设计模式学习-工厂方法模式
c++·学习·设计模式