如何实现is_enum_class类型特征?

Ink*_*ane 9 c++ metaprogramming sfinae type-traits

当且仅当传入的类型T是类枚举时,如何实现其值成员为true的类型特征?虽然我知道这一点

+T{};
Run Code Online (Sandbox Code Playgroud)

如果T是一个枚举,它将起作用,如果它是一个枚举类,则会失败,到目前为止我找不到将它用于SFINAE的方法.

Pio*_*cki 12

根据您的+T{}测试:

选项1:

尾随返回类型中的表达式SFINAE:

#include <type_traits>

template <typename T>
auto test(int) -> decltype((void)+T{}, std::false_type{});

template <typename T>
auto test(...) -> std::true_type;

template <typename T>
using is_enum_class = std::integral_constant<bool, decltype(test<T>(0))::value && std::is_enum<T>::value>;
Run Code Online (Sandbox Code Playgroud)

DEMO

选项#2:

void_t-fashion中:

template <typename T, typename V = void>
struct test : std::false_type {};

template <typename T>
struct test<T, decltype((void)+T{})> : std::true_type {};

template <typename T>
using is_enum_class = std::integral_constant<bool, !test<T>::value && std::is_enum<T>::value>;
Run Code Online (Sandbox Code Playgroud)

演示2

测试:

enum class EC { a, b };
enum E { c, d };

int main()
{
    static_assert(is_enum_class<EC>::value, "!");
    static_assert(!is_enum_class<E>::value, "!");
    static_assert(!is_enum_class<int>::value, "!");
}
Run Code Online (Sandbox Code Playgroud)


Col*_*mbo 6

您检查枚举是否可转换为int.

template <class T>
using is_scoped_enum = std::integral_constant<bool, !std::is_convertible<T,int>{}
                                                  && std::is_enum<T>{}>;
Run Code Online (Sandbox Code Playgroud)

这些静态断言将成功:

enum e {};
enum class e2 {};

static_assert( is_scoped_enum<e2>::value, "" );
static_assert( !is_scoped_enum<e>::value, "" );
static_assert( !is_scoped_enum<char>::value, "" );
Run Code Online (Sandbox Code Playgroud)

演示.

  • 我建议使用 `std::underlying_type&lt;T&gt;::type` 而不是 `int` (2认同)