C++11引入了两种不同的枚举处理方法:一个是让它们有作用域,另一个是让它们有类型。因此,现在我们有四种不同的枚举子类型:
enum Old {};
enum Typed : int8_t {};
enum class Scoped {};
enum class TypedScoped : int8_t {};
这个问题询问如何确定枚举是否有作用域。我想知道如何确定枚举是否有类型。
额外的信息
我使用Qt框架,它提供了QDataStream
类以便以一种便携式跨平台的方式序列化/反序列化数据。
显然,为了使结果数据流具有可移植性,您必须将所有整数存储在固定长度的形式中。这也包括枚举。
早些时候,我制作了几个辅助宏来定义枚举的序列化/反序列化,通过将它们转换为具有固定(用户指定)长度的整数:
#define SC_DECLARE_DATASTREAM_WRITE_OPERATOR(_TYPE) \
QDataStream &operator<<(QDataStream &stream, _TYPE v);
#define SC_DECLARE_DATASTREAM_READ_OPERATOR(_TYPE) \
QDataStream &operator>>(QDataStream &stream, _TYPE &v);
#define SC_DECLARE_DATASTREAM_OPERATORS(_TYPE) \
SC_DECLARE_DATASTREAM_WRITE_OPERATOR(_TYPE) \
SC_DECLARE_DATASTREAM_READ_OPERATOR(_TYPE)
#define SC_DEFINE_DATASTREAM_ENUM_WRITE_OPERATOR(_TYPE, _LEN) \
QDataStream &operator<<(QDataStream &stream, _TYPE v) \
{ \
qint ## _LEN t = v; \
static_assert(sizeof(t) >= sizeof(v), "Increase length"); \
stream << t; \
return stream; \
}
#define SC_DEFINE_DATASTREAM_ENUM_READ_OPERATOR(_TYPE, _LEN) \
QDataStream &operator>>(QDataStream &stream, _TYPE &v) \
{ \
qint ## _LEN t {0}; \
static_assert(sizeof(t) >= sizeof(v), "Increase length"); \
stream >> t; \
if(stream.status() == QDataStream::Ok) \
v = static_cast<_TYPE>(t); \
return stream; \
}
#define SC_DEFINE_DATASTREAM_ENUM_OPERATORS(_TYPE, _LEN) \
SC_DEFINE_DATASTREAM_ENUM_WRITE_OPERATOR(_TYPE, _LEN) \
SC_DEFINE_DATASTREAM_ENUM_READ_OPERATOR(_TYPE, _LEN)
现在,C++11允许指定底层枚举类型,因此我可以简化上述提到的宏:
#define SC_DEFINE_DATASTREAM_TYPED_ENUM_WRITE_OPERATOR(_TYPE) \
QDataStream &operator<<(QDataStream &stream, _TYPE v) \
{ \
const std::underlying_type<_TYPE>::type t {static_cast<std::underlying_type<_TYPE>::type>(v)}; \
stream << t; \
return stream; \
}
#define SC_DEFINE_DATASTREAM_TYPED_ENUM_READ_OPERATOR(_TYPE) \
QDataStream &operator>>(QDataStream &stream, _TYPE &v) \
{ \
std::underlying_type<_TYPE>::type t {0}; \
stream >> t; \
if(stream.status() == QDataStream::Ok) \
v = static_cast<_TYPE>(t); \
return stream; \
}
但是,如果用户不小心在没有指定底层类型的枚举中使用新的(*_TYPED_*
)宏,那么这将打破可移植性的保证,因为在不同平台上编译相同代码可能会产生不同的底层类型,从而导致序列化/反序列化代码中的整数长度不同。
我需要添加一个 static_assert
到代码中,如果枚举在声明时没有强类型化,它将会打破编译过程。
enum Old { VALUE = -1ULL; };
是一种带类型的枚举吗? - NathanOliverenum
是否具有指定类型,但是你可以检查其底层类型是什么:https://en.cppreference.com/w/cpp/types/underlying_type - Ben Jonesint
只适用于有作用域的枚举。 - NathanOliverchar
到int
的类型都是可接受的。enum foo { bar };
合法地允许具有char
底层类型。 - NathanOliver