Как я могу исправить эту ошибку компиляции шаблона списка типов c ++? - PullRequest
0 голосов
/ 28 января 2010

(из прочтения главы 3 современного дизайна c ++)

typelist.hpp:

class NullType {};

struct EmptyType {};


template <class T, class U>
struct Typelist
{
  typedef T Head;
  typedef U Tail;
};

#define TYPELIST_1(T1) Typelist<T1, NullType>
#define TYPELIST_2(T1, T2) Typelist<T1, TYPELIST_1(T2) >
#define TYPELIST_3(T1, T2, T3) Typelist<T1, TYPELIST_2(T2, T3) >
#define TYPELIST_4(T1, T2, T3, T4) Typelist<T1, TYPELIST_3(T2, T3, T4) >
#define TYPELIST_5(T1, T2, T3, T4, T5) Typelist<T1, TYPELIST_4(T2, T3, T4, T5) >
#define TYPELIST_6(T1, T2, T3, T4, T5, T6) Typelist<T1, TYPELIST_5(T2, T3, T4, T5, T6) >


namespace TL
{
  template <class TList> struct Length;
  template <> struct Length<NullType>
  {
    enum { value = 0 };
  };

  template <class T, class U>
    struct Length< Typelist<T, U> >
    {
      enum { value = 1 + Length<U>::value };
    };


  template <class Head, class Tail>
    struct TypeAt<Typelist<Head, Tail>, 0>
    {
      typedef Head Result;
    };

  template <class Head, class Tail, unsigned int i>
    struct TypeAt<Typelist<Head, Tail>, i>
    {
      typedef typename TypeAt<Tail, i-1>::Result Result;
    };

}

main.cpp

#include "typelist.hpp"

Typelist<int, double> foo;

int main() {
}

g ++ main.cpp

typelist.hpp:37: error: ‘TypeAt’ is not a template
typelist.hpp:43: error: type/value mismatch at argument 2 in template parameter list for ‘template<class Head, class Tail> struct TL::TypeAt’
typelist.hpp:43: error:   expected a type, got ‘i’

Почему я получаю эту ошибку? Как мне это исправить?

Ответы [ 2 ]

3 голосов
/ 28 января 2010

Похоже, вы пропустили предварительную декларацию.

Это частичная специализация:

template <class Head, class Tail>
struct TypeAt<Typelist<Head, Tail>, 0>

Но компилятор понятия не имеет, что это за специализация. Добавьте это перед этим:

template <class List, unsigned Index>
struct TypeAt;

Это даст знать компилятору: «Существует класс TypeAt, который имеет два параметра шаблона». Так что теперь, когда вы специализируете его, компилятор знает, о каком классе вы говорите.


Обратите внимание, что вы используете Typelist неправильно. Эти алгоритмы заканчиваются на страже . Это означает, что, как и C-строки, они ожидают, что данные будут заключены с особым значением. В нашем случае это NullType.

Итак, примите совет Эрика . (т.е. подсказка: если вы нашли его ответ полезным, проголосуйте за него.)

1 голос
/ 28 января 2010

Typelist предназначен для рекурсивного использования: он ожидает, что его второй параметр-шаблонатор будет либо другим Typelist, либо NullType (сигнализирующим об окончании рекурсии).

Чтобы определить foo, вы должны написать:

TYPELIST_2(int, double) foo;

или, если вы хотите избежать макросов:

Typelist<int, Typelist<double, NullType> > foo;
...