真正的问题是你需要一个C++ 03解决方案,所以你可以使用SfEAE,但不是所有的语言改进都可以从C++ 11开始。
不管怎样,我建议你一个解决方案,是复杂的像你的(也许更多),但完全没有刺激。
如果定义了一个普通的BoL包装器(它可以粗略地取代C++ 11)
std::true_type
std::false_type
)
template <bool B>
struct bool_wrapper
{ static const bool value = B; };
template <typename, typename = bool_wrapper<true> >
struct cond : public bool_wrapper<false>
{ };
template <typename T>
struct cond<T, bool_wrapper<(1 == T::code) || (2 == T::code)> >
: public bool_wrapper<true>
{ };
如果你定义一个
enable_if
类型特征(与C++ 11相同)
std::enable_if
template <bool, typename = void>
struct enable_if
{ };
template <typename T>
struct enable_if<true, T>
{ typedef T type; };
您可以启用/禁用
foo()
功能
template <typename T>
typename enable_if<false == cond<T>::value>::type foo (T const & arg)
{ std::cout << "no static member code or value not 1 and not 2\n"; }
template <typename T>
typename enable_if<true == cond<T>::value>::type foo (T const & arg)
{ std::cout << "static member code and its value is " << T::code << "\n"; }
#include <iostream>
template <int N> struct A1 { static const int code = N; };
template <int N> struct A2 { static const int code = N; };
// ...
template <int N> struct AN { static const int code = N; };
struct B1{};
struct B2{};
// ...
struct BN{};
template <bool B>
struct bool_wrapper
{ static const bool value = B; };
template <typename, typename = bool_wrapper<true> >
struct cond : public bool_wrapper<false>
{ };
template <typename T>
struct cond<T, bool_wrapper<(1 == T::code) || (2 == T::code)> >
: public bool_wrapper<true>
{ };
template <bool, typename = void>
struct enable_if
{ };
template <typename T>
struct enable_if<true, T>
{ typedef T type; };
template <typename T>
typename enable_if<false == cond<T>::value>::type foo (T const & arg)
{ std::cout << "no static member code or value not 1 and not 2\n"; }
template <typename T>
typename enable_if<true == cond<T>::value>::type foo (T const & arg)
{ std::cout << "static member code and its value is " << T::code << "\n"; }
int main ()
{
foo(A1<0>()); // match the 1st version of foo
foo(A2<1>()); // match the 2nd version of foo
foo(AN<2>()); // match the 2nd version of foo
foo(B1()); // match the 1st version of foo
foo(BN()); // match the 1st version of foo
}
我不知道您使用的boost类,但我想您可以修改您的代码(使其几乎与我的无boost解决方案一样工作),如下所示
template <typename, typename = bool_<true> >
struct Condition : public bool_<false>
{ };
template <typename T>
struct Condition<T, bool_<(1 == T::code) || (2 == T::code)> >
: public bool_<true>
{ };
--编辑--
行动询问
我不明白它如何适用于案例A1<0>。条件的专门化应该是首选匹配,第二个参数扩展到bool。该类继承自bool,因此,它应该选择错误的foo版本。不管它是怎么工作的。怎么可能?
好。。。当你写作的时候
foo(A1<0>())
,编译器必须了解
cond<A1<0>>::value
true
或
false
启用的第一个版本
foo()
或者第二个。
cond<A1<0>>
. 但是没有一个
cond
只接收类型名的模板类。不管怎样,编译器发现
template <typename, typename = bool_wrapper<true> >
struct cond;
使用第二个模板参数的默认值进行匹配。
cond< A<1> >
cond< A<1>, bool_wrapper<true> >
现在编译器必须在
cond<typename, typename>
(继承自
bool_wrapper<false>
bool_wrapper<true>
).
当然匹配主版本,但也匹配专门化?如果匹配也匹配专门化,编译器必须首选专门化。
cond< A<0>, bool_wrapper<true> >
与专业化相匹配。
使用
A<0>
T
cond< A<0>, bool_wrapper<(1 == A<0>::code) || (2 == A<0>::code)> >
就是这样
cond< A<0>, bool_wrapper<(1 == 0) || (2 == 0)> >
cond< A<0>, bool_wrapper<false || false> >
就是这样
cond< A<0>, bool_wrapper<false> >
这不匹配
.
所以呢
cond< A<0> >
,即
,仅与的主版本匹配
条件(&L);typename,typename>
bool\u包装器<错误>
.
.
至于
条件(&L);<0>&燃气轮机;
条件
条件(&L);<1>&燃气轮机;
是
与第二个
typename
使用默认值。
所以呢
是
但是
条件(&L);<1>,bool\u包装器<正确>&燃气轮机;
条件(&L);typename,typename>
还是专业化?
我们可以看到
A<1>
作为
,专业化成为
cond< A<1>, bool_wrapper<(1 == A<1>::code) || (2 == A<1>::code)> >
就是这样
cond< A<1>, bool_wrapper<(1 == 1) || (2 == 1)> >
就是这样
cond< A<1>, bool_wrapper<true || false> >
就是这样
cond< A<1>, bool_wrapper<true> >
条件(&L);<1>,bool\u包装器<正确>&燃气轮机;
.
条件(&L);<1>&燃气轮机;
cond< A<1>, bool_wrapper<true>
,两个版本的
条件(&L);typename,typename>
匹配,所以编译器必须选择专门化,所以
条件(&L);<1>&燃气轮机;
继承自
bool\u包装器<正确>
.