使用std :: underlying_type

时间:2016-04-12 08:48:01

标签: c++ enums variadic-templates sfinae overload-resolution

下面的SFINAE代码使用可变参数模板很好地编译使用clang 3.7.1,C ++ 14:

#include <array>
#include <iostream>
#include <vector>
#include <cstdint>

enum class Bar : uint8_t {
    ay, bee, see
};

struct S {

static void foo() {}

// std::begin(h) is defined for h of type H
template<typename H, typename... T>
static typename std::enable_if<std::is_pointer<decltype(std::begin(std::declval<H>()))*>::value>::type 
foo(const H&, T&&... t) 
{ std::cout << "container\n"; foo(std::forward<T>(t)...); }

// H is integral
template<typename H, typename... T>
static typename std::enable_if<std::is_integral<typename std::remove_reference<H>::type>::value>::type 
foo(const H&, T&&... t) 
{ std::cout << "integer\n"; foo(std::forward<T>(t)...); }

// H is an enum with underlying type = uint8_t
/*
template<typename H, typename... T>
static typename std::enable_if<std::is_same<typename std::underlying_type<H>::type,uint8_t>::value>::type 
foo(const H&, T&&... t)
{ std::cout << "enum\n"; foo(std::forward<T>(t)...); }
*/
};


int main()
{
    S::foo(std::array<int,8>(), 5, 5L, std::vector<int>{}, 5L);
}

我希望根据foo类型递归调用H的正确重载:

  1. 如果为std::begin(h)类型h定义H,我想要。{ 过载数字1将被选择
  2. 如果H是“整数类型”,我想要重载2号。
  3. 这样可行。但是如果我为 enum 类型添加另一个重载(你可以尝试取消注释第三个重载),那么我得到:

      

    错误:只有枚举类型具有基础类型

    我同意只有 enums 有一个基础类型,因此为什么不是第三次重载(std::underlying_type)得到SFINAE-d?

2 个答案:

答案 0 :(得分:17)

std::underlying_type不是SFINAE友好的。尝试访问非枚举类型的std::underlying_type<T>::type会导致未定义的行为(通常是硬错误),而不是替换失败。

在尝试访问其基础类型之前,您需要首先确定所讨论的类型是枚举类型。排成一行就像typename std::enable_if<std::is_enum<H>::value, std::underlying_type<H>>::type::type一样。用这个可怕的混乱替换你的返回类型中的typename std::underlying_type<H>::type,你会得到一个更加可怕的混乱:)

如果您发现自己经常需要这样做 - 或者只是不想写typename std::enable_if<std::is_same<typename std::enable_if<std::is_enum<H>::value, std::underlying_type<H>>::type::type, uint8_t>::value>::type - 您可以写一个SFINAE友好的underlying_type

template<class T, bool = std::is_enum<T>::value>
struct safe_underlying_type : std::underlying_type<T> {};
template<class T>
struct safe_underlying_type<T, false /* is_enum */> {};

答案 1 :(得分:1)

这是一个受T.C.'s solution启发的解决方案,适用于我的用例:

template <typename T, bool = std::is_enum<T>::value>
struct relaxed_underlying_type {
    using type = typename std::underlying_type<T>::type;
};

template <typename T>
struct relaxed_underlying_type<T, false> {
    using type = T;
};

示例用法:

template <typename T>
struct UnwrapEnum {
    using type =
        typename std::conditional<
        std::is_enum<T>::value,
        typename relaxed_underlying_type<T>::type,
        T>
        ::type;
};

enum class MyEnum : int {};

class MyClass {};

int main() {
    UnwrapEnum<MyEnum>::type x;
    static_assert(std::is_same<decltype(x), int>::value);

    UnwrapEnum<MyClass>::type y;
    static_assert(std::is_same<decltype(y), MyClass>::value);

    return 0;
}