c++ - 返回枚举的基础类型和迭代器的整数的元编程构造
问题描述
我需要编写一个元编程构造,当给定枚举类型时,它返回该枚举的基础类型,但是当给定整数时,它返回该整数。
例如:
enum Enum : short { VALUE1, VALUE2 };
int_type<long>::type // -> long
int_type<Enum>::type // -> short
我试过这个
template< typename Type >
struct int_type {
using type = typename std::enable_if< std::is_enum<Type>::value, typename std::underlying_type<Type>::type >::type;
};
template< typename Type >
struct int_type {
using type = typename std::enable_if< std::is_integral<Type>::value, Type >::type;
};
但它抱怨结构的重新定义。
我也试过这个,但是
template< typename Type >
struct int_type {
using type = typename std::enable_if< std::is_enum<Type>::value, typename std::underlying_type<Type>::type >::type;
using type = typename std::enable_if< std::is_integral<Type>::value, Type >::type;
};
但随后它抱怨重新定义 member type
。
这就是我的元编程技能结束的地方,有人可以帮忙吗?
编辑:我还应该提到,我们的项目仅限于 C++11。
解决方案
您尝试的问题是您定义了两次相同的东西,虽然它的一部分使用它并不会在不正确enable_if
时禁用整个封闭定义。enable_if
即,int_type::type
即使其中一个(或两个)定义无效,您也有两个定义。
您正在寻找的是std::conditional
:
template< typename Type >
struct int_type {
using when_enum
= std::underlying_type<Type>;
using when_integral
= std::enable_if<std::is_integral<Type>::value, Type>;
using type
= typename std::conditional< std::is_enum<Type>::value,
when_enum, when_integral
>::type::type;
};
或者在 C++14 及更高版本中:
template< typename Type >
struct int_type {
using when_enum
= std::underlying_type<Type>;
using when_integral
= std::enable_if<std::is_integral<Type>::value, Type>;
using type
= typename std::conditional_t< std::is_enum<Type>::value,
when_enum, when_integral
>::type;
};
但我想我可能会把它写成别名模板:
template< typename Type >
using int_type = std::conditional_t<
std::is_enum<Type>::value,
std::underlying_type<Type>,
std::enable_if<std::is_integral<Type>::value, Type>>;
也许进一步简化它并int_type<T>
为实际类型创建一个别名,所以你不需要说typename int_type<T>::type
:
template< typename Type >
using int_type = typename std::conditional_t<
std::is_enum<Type>::value,
std::underlying_type<Type>,
std::enable_if<std::is_integral<Type>::value, Type>>::type;
(注意在 C++17 中,您可以使用std::is_enum_v<Type>
andstd::is_integral_v
而不是说is_xxx<Type>::value
.)