Специализация шаблона на основе класса inherit
Я хочу сделать это специализированным без изменения основного. Можно ли специализироваться на своем базовом классе? Надеюсь, что так.
-edit -
У меня будет несколько классов, которые наследуют от SomeTag. Я не хочу писать одну и ту же специализацию для каждого из них.
class SomeTag {};
class InheritSomeTag : public SomeTag {};
template <class T, class Tag=T>
struct MyClass
{
};
template <class T>
struct MyClass<T, SomeTag>
{
typedef int isSpecialized;
};
int main()
{
MyClass<SomeTag>::isSpecialized test1; //ok
MyClass<InheritSomeTag>::isSpecialized test2; //how do i make this specialized w/o changing main()
return 0;
}
Ответы
Ответ 1
В этой статье описывается аккуратный трюк: http://www.gotw.ca/publications/mxc++-item-4.htm
Вот основная идея. Сначала вам нужен класс IsDerivedFrom (это обеспечивает проверку времени выполнения и время компиляции):
template<typename D, typename B>
class IsDerivedFrom
{
class No { };
class Yes { No no[3]; };
static Yes Test( B* ); // not defined
static No Test( ... ); // not defined
static void Constraints(D* p) { B* pb = p; pb = p; }
public:
enum { Is = sizeof(Test(static_cast<D*>(0))) == sizeof(Yes) };
IsDerivedFrom() { void(*p)(D*) = Constraints; }
};
Затем ваш MyClass нуждается в реализации, которая потенциально специализируется:
template<typename T, int>
class MyClassImpl
{
// general case: T is not derived from SomeTag
};
template<typename T>
class MyClassImpl<T, 1>
{
// T is derived from SomeTag
public:
typedef int isSpecialized;
};
и MyClass выглядит так:
template<typename T>
class MyClass: public MyClassImpl<T, IsDerivedFrom<T, SomeTag>::Is>
{
};
Тогда ваш основной будет хорошо, как это:
int main()
{
MyClass<SomeTag>::isSpecialized test1; //ok
MyClass<InheritSomeTag>::isSpecialized test2; //ok also
return 0;
}
Ответ 2
Ну, статья в ответе выше появилась в феврале 2002 года. Пока она работает, сегодня мы знаем, что есть лучшие способы. Кроме того, вы можете использовать enable_if
:
template<bool C, typename T = void>
struct enable_if {
typedef T type;
};
template<typename T>
struct enable_if<false, T> { };
template<typename, typename>
struct is_same {
static bool const value = false;
};
template<typename A>
struct is_same<A, A> {
static bool const value = true;
};
template<typename B, typename D>
struct is_base_of {
static D * create_d();
static char (& chk(B *))[1];
static char (& chk(...))[2];
static bool const value = sizeof chk(create_d()) == 1 &&
!is_same<B volatile const,
void volatile const>::value;
};
struct SomeTag { };
struct InheritSomeTag : SomeTag { };
template<typename T, typename = void>
struct MyClass { /* T not derived from SomeTag */ };
template<typename T>
struct MyClass<T, typename enable_if<is_base_of<SomeTag, T>::value>::type> {
typedef int isSpecialized;
};
int main() {
MyClass<SomeTag>::isSpecialized test1; /* ok */
MyClass<InheritSomeTag>::isSpecialized test2; /* ok */
}
Ответ 3
И короткая версия сейчас, 2014, используя С++ - 11:
#include <type_traits>
struct SomeTag { };
struct InheritSomeTag : SomeTag { };
template<typename T, bool = std::is_base_of<SomeTag, T>::value>
struct MyClass { };
template<typename T>
struct MyClass<T, true> {
typedef int isSpecialized;
};
int main() {
MyClass<SomeTag>::isSpecialized test1; /* ok */
MyClass<InheritSomeTag>::isSpecialized test2; /* ok */
}
Ответ 4
В вашем случае единственный способ, который я вижу, - явно указать MyClass
для InheritSomeTag
. Тем не менее, документ SeqAn предлагает механизм, называемый "затенение шаблона", который делает то, что вы хотите, хотя и с другим синтаксисом наследования, поэтому код isn 't совместим с вашей текущей функцией main
.
// Base class
template <typename TSpec = void>
class SomeTag { };
// Type tag, NOT part of the inheritance chain
template <typename TSpec = void>
struct InheritSomeTag { };
// Derived class, uses type tag
template <typename TSpec>
class SomeTag<InheritSomeTag<TSpec> > : public SomeTag<void> { };
template <class T, class Tag=T>
struct MyClass { };
template <class T, typename TSpec>
struct MyClass<T, SomeTag<TSpec> >
{
typedef int isSpecialized;
};
int main()
{
MyClass<SomeTag<> >::isSpecialized test1; //ok
MyClass<SomeTag<InheritSomeTag<> > >::isSpecialized test2; //ok
}
Это, конечно, выглядит странно и очень громоздко, но позволяет использовать механизм наследования с полиморфными функциями, выполняемый во время компиляции. Если вы хотите увидеть это в действии, посмотрите на Примеры SeqAn.
Считаю, что SeqAn - это особый случай, и не многие приложения выиграют от этого чрезвычайно сложного синтаксиса (расшифровка связанных с SeqAn ошибок компилятора - настоящая боль в * ss!)