几个级别的嵌套模板。我如何让它工作?

时间:2014-04-04 22:37:44

标签: c++ templates template-meta-programming typetraits

我正在做一些模板元编程,我有这样的情况,首先我有几个类,如: -

template <typename Q>
struct Object {
 public:
  Q data;
};

template <typename P>
class CircleObject : public Object<const typename P::Circle> {
};

template <typename P>
class SquareObject : public Object<const typename P::Circle> {
};

template <typename P>
class Storage {
 public:
  typedef CircleObject<P> MyCircle;
  typedef SquareObject<P> MySquare;
};

现在,我试图定义这些对象的一些特征: -

template <typename P>
struct CircleTraits<Storage<P> > {
  template <typename otype>
  struct IsCircle {
    static const bool VALUE = false;
  };
};

template <typename P>
struct CircleTraits<Storage<P> >::IsCircle<Storage<P>::MyCirlce> {
  static const bool VALUE = true;
};

但是,这是不正确的(编译错误)。我已经尝试过将测试名称和模板参数放在任何地方的试错法,但是如果没有对模板专业化的深刻理解,我真的无法解决这个问题。有人可以在这帮忙吗?

我希望在以后的功能中实现的目标如下: -

typedef Storage<RedObjects> RedStorage;

template <typename SpecializedStorage>
class Processor {
  typedef CircleTraits<typename SpecializedStorage> MyCircleTraits;

  template <typename ObjectType>
  void foo(ObjectType& data);
};

template <typename SpecializedStorage>
template <typename ObjectType>
void foo(ObjectType& data) {
  if (MyCircleTraits::template IsCircle<ObjectType>::VALUE) {
    // do something about the damn circles
  }
}

1 个答案:

答案 0 :(得分:1)

我认为你不能这样做,你可能应该使用SFINAE来解决这样的问题:

//C++11 version
template<typename T>
struct IsCircle
{
private:
    template<typename Z>
    constexpr static bool _is(typename Z::MyCirlce*) //if Z dont have `MyCirlce` then this function is removed
    {
        return true;
    }
    template<typename Z>
    constexpr static bool _is(...) //fallback function
    {
        return false;
    }
public:
    static const bool VALUE = _is<T>(nullptr);
};

//C++98 version
template<typename T>
struct IsCircle
{
private:
    struct a { char a; }; //size ~1
    struct b { char a[8]; }; //size ~8
    template<typename Z>
    static b _is(typename Z::MyCirlce*);
    template<typename Z>
    static a _is(...);
public:
    static const bool VALUE = sizeof(_is<T>(0)) == sizeof(b);
};