C ++:使用CRTP,派生类中定义的类在基类中不可访问

时间:2011-04-04 05:06:10

标签: c++ qt templates crtp

这是(简化的)基类:

template <class T>
class SharedObject
{
protected:
    QExplicitlySharedDataPointer <typename T::Data> d;
};

这是派生的:

class ThisWontCompile : public SharedObject <ThisWontCompile>
{
private:
    friend class SharedObject;
    struct Data : public QSharedData
    {
        int id;
    };
};

是否有任何解决方法可以从 SharedObject 访问 ThisWontCompile :: Data ?使用基础对象中的派生对象究竟能做什么,究竟什么不能做什么呢?

1 个答案:

答案 0 :(得分:13)

这实际上与可访问性和友谊无关,它与CRTP的使用有关。请考虑以下示例,该示例也会出现问题:

template <class T>
struct Base
{
    typedef typename T::Data Data;
};

struct ThisWontCompile : public Base<ThisWontCompile>
{
    struct Data { };
};

问题是ThisWontCompile在用作Base的模板参数时不完整,因此它只能用作Base中的不完整类型。

针对您的具体问题的一些解决方案,请参阅this other question的答案,特别是Martin建议使用特征类,基本上如下:

// Base
template <typename T>
struct BaseTraits;

template <typename T>
struct Base
{
    typedef typename BaseTraits<T>::Data Data;
};

// Derived
struct Derived;

template <>
struct BaseTraits<Derived>
{
    struct Data { };
};

struct Derived : public Base<Derived>
{
};

typename BaseTraits<Derived>::Data可以在DerivedBase中使用。如果Derived本身就是一个模板,则可以对traits类使用部分特化。