كيف يمكنني إصلاح خطأ ترجمة قالب Typelist C ++؟
سؤال
(من القراءة الفصل 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’
لماذا أحصل على هذا الخطأ؟ كيف يمكنني حل هذه المشكلة؟
المحلول
يبدو أنك تفتقد إعلانًا إلى الأمام.
هذا تخصص جزئي:
template <class Head, class Tail>
struct TypeAt<Typelist<Head, Tail>, 0>
لكن المترجم ليس لديه أي فكرة عما هو تخصص. أضف هذا قبل ذلك:
template <class List, unsigned Index>
struct TypeAt;
هذا يدع المترجم يعرف: "هناك فئة TypeAt
التي تحتوي على معلمتان للقالب. "حتى الآن عندما تتخصص ذلك ، يعرف المترجم الفئة التي تتحدث عنها.
ملاحظة ، استخدامك من Typelist
غير صحيح. هذه الخوارزمية المنتهية الحارس. هذا يعني ، مثل السلاسل C ، يتوقعون أن يتم الانتهاء من البيانات بقيمة خاصة. في حالتنا ، هذا هو NullType
.
لذلك ، خذ éricنصيحة. (أي تلميح: إذا وجدت إجابته مفيدة ، فهي صاخبة.)
نصائح أخرى
تم تصميم TypElist لاستخدامه بشكل متكرر: فهو يتوقع أن تكون معلمة Templater الثانية إما طباعة أخرى ، أو nulltype (تشير إلى نهاية العودية).
لتعريف فو ، يجب أن تكتب:
TYPELIST_2(int, double) foo;
أو ، إذا كنت تريد تجنب وحدات الماكرو:
Typelist<int, Typelist<double, NullType> > foo;