C ++ 11类型特征来区分枚举类和常规枚举

我正在编写一个类似于boost :: promote的升级模板别名,但是用于C ++ 11。 这样做的目的是在从varidic函数中检索参数时避免警告。 例如

template <typename T>
std::vector<T> MakeArgVectorV(int aArgCount, va_list aArgList)
{
    std::vector<T> args;
    while (aArgCount > 0)
    {
        args.push_back(static_cast<T>(va_arg(aArgList, Promote<T>)));
        --aArgCount;
    }
    return args;
}

Promote模板别名在可变参数的默认参数提升之后提升类型:1)小于int的整数提升为int 2)float提升为double

我的问题是,可以提升标准C ++枚举,但不推荐C ++ 11枚举类(编译器不会生成警告)。 我想促进与常规枚举一起工作,但忽略C ++ 11枚举类。

我如何分辨我的Promote模板别名中的枚举类和枚举之间的区别?


这是一个可能的解决方案:

#include <type_traits>

template<typename E>
using is_scoped_enum = std::integral_constant<
    bool,
    std::is_enum<E>::value && !std::is_convertible<E, int>::value>;

该解决方案利用了C ++ 11标准第7.2 / 9段中规定的范围和非范围枚举之间的行为差​​异:

枚举器或非范围枚举类型的对象的值通过整数提升(4.5)转换为整数。 [...]请注意,这种隐式枚举到int转换不是为范围枚举提供的。 [...]

以下是您将如何使用它的演示:

enum class E1 { };
enum E2 { };
struct X { };

int main()
{
    // Will not fire
    static_assert(is_scoped_enum<E1>::value, "Ouch!");

    // Will fire
    static_assert(is_scoped_enum<E2>::value, "Ouch!");

    // Will fire
    static_assert(is_scoped_enum<X>::value, "Ouch!");
}

这里是一个生动的例子。

致谢:

感谢Daniel Frey指出,只要没有用户定义的operator +超载,我以前的方法就会工作。

链接地址: http://www.djcxy.com/p/90829.html

上一篇: C++11 type trait to differentiate between enum class and regular enum

下一篇: Are "anonymous structs" standard? And, really, what *are* they?