#include <type_traits>
#define str_cat(first, second) first##second
#define has_xxx(member_name) \
template<class T, class U = void> \
struct str_cat(has_, member_name): std::false_type {};\
template<class T>\
struct str_cat(has_, member_name)<T, typename SFINAE<typename T::member_name>::type>\
: std::true_type {};\
template<class T>\
struct str_cat(has_, member_name)<T, typename SFINAE<decltype(T::member_name)>::type>\
: std::true_type {};
template<class T>
struct SFINAE {
using type = void;
};
struct A {
int i = 0;
void foo() {}
using void_t = void;
};
struct B {
int j = 0;
void goo() {}
using void_t = void;
};
has_xxx(i)
has_xxx(foo)
has_xxx(j)
has_xxx(goo)
has_xxx(void_t) //compile error if `has_xxx(i)` appears at the head
int main()
{
//has_i<A>::value; // true
//has_i<B>::value; // false
has_foo<A>::value; // true
has_foo<B>::value; // false
has_goo<B>::value; // true
has_void_t<A>::value; // true
has_void_t<B>::value; // true
return 0;
}
На VS2017 он не компилируется
https://gcc.godbolt.org/z/JkOhLi
ошибка: C2752 'template': более одной частичной специализации соответствует списку аргументов шаблона.
Но это нормально для gcc и clang. http://coliru.stacked -crooked.com / a / 6b9490f6b127ae88
Если я изменяю порядок макроса, он компилируется:
has_xxx(foo)
has_xxx(i) //now compiles successfully
has_xxx(j)
has_xxx(goo)
has_xxx(void_t) //compile error if `has_xxx(i)` appears at the head
Или просто измените имя члена в структуре A:
struct A {
int k = 0; // i -> k, now compiles successfully !!!!
void foo() {}
using void_t = void;
};
Я не могу понять причину.
Имеет ли значение порядок макросов или это ошибка MSVC на SFINAE?