【问题标题】:Statically distinguish between an enum and an enum class in C++?静态区分 C++ 中的枚举和枚举类?
【发布时间】:2020-06-15 10:11:04
【问题描述】:

我有一个事件处理程序类,它使用模板参数来设置事件类型。我想将这些事件类型强制为一个字节大小的 枚举类。针对大小的静态断言不是问题,但我无法在网上找到有关如何静态区分 enumenum 类 的信息。

我现在的解决方案是使用C++ front-end implements syntactic extensions 为枚举断言,然后断言正确的大小。在大多数平台上这是可行的,因为枚举使用 int 类型(通常大于一个字节)。

但这会导致稍微误导性的错误消息。我喜欢彻底。

我可以做哪些检查才能通过类枚举,但会因常规旧枚举而失败?


我不能使用type_traits,因为我使用的编译器 (avr-gcc) 不支持它。然而,当需要时,我会不断地实施我自己的type_traits。所以type_traits 中有关解决方案的任何提示仍然有用!


简单示例:

// Event types
enum class tPass     : uint8_t  {}; 
enum class tFailSize : uint16_t {}; // Fail on size!
enum       tFailType            {}; // Currently fails on size, would like to fail on type!

// Event handler
template <typename TEvent>
class tEventHandler 
{
    static_assert(__is_enum(TEvent),   "Must be class enum!"); // Doesn't really check for CLASS enums
    static_assert(1 == sizeof(TEvent), "Must be one byte!");
};

用法:

auto handler = tEventHandler<tPass>();   // Ok!
// auto handler2 = tEventHandler<tFailSize>(); // error: static assertion failed: Must be one byte!
// auto handler3 = tEventHandler<tFailType>(); // error: static assertion failed: Must be one byte! <----- THIS

目标:

auto handler = tEventHandler<tPass>();   // Ok!
// auto handler2 = tEventHandler<tFailSize>(); // error: static assertion failed: Must be one byte!
// auto handler3 = tEventHandler<tFailType>(); // error: static assertion failed: Must be class enum! <--- THIS

【问题讨论】:

    标签: c++ enums enumeration static-assert enum-class


    【解决方案1】:

    无作用域枚举和作用域类型 (enum class) 之间的一个关键区别在于,前者隐式转换为其基础类型,而后者则不会。由于您对type_traits 解决方案的外观感兴趣,您可以查看(未完全测试,可能不适合 SFINAE):

    static_assert(std::is_enum<TEvent>::value, "Must be a scoped enum!"); 
    static_assert(!std::is_convertible<TEvent, typename std::underlying_type<TEvent>::type>::value,
                  "Must be a scoped enum!");
    

    如果 AVR-GCC 具有 underlying_type 内在特性,那么您只需补充 is_convertible 特征(甚至可以在 C++03 中实现)。

    【讨论】:

      猜你喜欢
      • 2015-04-26
      • 2013-03-13
      • 2014-06-01
      • 1970-01-01
      • 2010-11-26
      • 1970-01-01
      • 2016-01-02
      • 1970-01-01
      • 2013-10-27
      相关资源
      最近更新 更多