带有宏的类中的 C++ 自动 shared_ptr 类型

问题描述

我正在尝试为我的项目中的每个类添加一个 shared_ptr 的类内别名,如下所示:

class Foo {
/* ... */
public:
    using Ptr = std::shared_ptr<Foo>;
};

这样我就可以用简写 Foo::Ptr fooObjPtr; 定义指向 Foo 的共享指针。有没有什么方法可以创建一个自动添加别名的宏?类似的东西:

#define DEFINE_SHARED using Ptr = std::shared_ptr<__CLASS_NAME__>;

class Foo {
/* ... */
public:

    DEFINE_SHARED
};

解决方法

类模板可以做到这一点:

template<typename T>
class FooBase
{
    public:
        using Ptr = std::shared_ptr<T>;
};

class Foo :
    public FooBase<Foo>
{
};

int main()
{
    Foo::Ptr x = std::make_shared<Foo>();
    
    std::cout << x << std::endl;
}

这应该可以实现您的要求,而无需依赖任何预处理器功能。

请注意,根据您的用例,您可能需要添加一些语法糖,例如确保 FooBase::T 实际上继承自 FooBase。有几种解决方案 - 查找 CRTP,因为这是一个常见的“问题”。

,
#define CreateClass(classname) class classname { \
    using Ptr = std::shared_ptr<classname>;

#define CreateStruct(classname) struct classname { \
    using Ptr = std::shared_ptr<classname>;

这很丑陋,但可以解决问题。

示例:

class MyClass {
    using Ptr = std::shared_ptr<MyClass>;

    virtual ~MyClass() = default;

    /* .... */
};

// Equivalent to:

CreateClass(MyClass)

    virtual ~MyClass() = default;

    /* .... */

};

@编辑

稍微扩展宏,可以添加基类:

#define _ClassCreation1(classname) class classname { \
    using Ptr = std::shared_ptr<classname>;

#define  _ClassCreation2(classname,baseclasses) class classname : baseclasses { \
    using Ptr = std::shared_ptr<classname>;

#define _ARG2(_0,_1,_2,...) _2
#define NARG2(...) _ARG2(__VA_ARGS__,2,1,0)

#define _ONE_OR_TWO_ARGS_1(a) _ClassCreation1(a)
#define _ONE_OR_TWO_ARGS_2(a,b) _ClassCreation2(a,b)

#define __ONE_OR_TWO_ARGS(N,...) _ONE_OR_TWO_ARGS_ ## N (__VA_ARGS__)
#define _ONE_OR_TWO_ARGS(N,...) __ONE_OR_TWO_ARGS(N,__VA_ARGS__)

#define CreateClass(...) _ONE_OR_TWO_ARGS(NARG2(__VA_ARGS__),__VA_ARGS__)

(基于:https://stackoverflow.com/a/11763196/5589708

所以你会像这样使用它:

CreateClass(MyClass)

    virtual ~MyClass() = default;

    /* .... */
};

CreateClass(MyClass2,public MyClass)

    virtual ~MyClass2() = default;

    /* .... */

};