是否可以确定枚举是否为强类型?
Is it possible to determine if an enumeration was strongly typed?
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
,如果枚举在其声明点没有强类型,这将破坏编译过程。
std::underlying_type
可用于将编译限制为一组fixed width integer types
(例如,使用std::is_same
(:
#include <type_traits>
#include <cstdint>
template <typename T>
constexpr bool is_fixed =
std::is_same<T, std::int8_t>::value ||
std::is_same<T, std::int16_t>::value
// etc..
;
enum class E1 : std::int8_t {};
static_assert( is_fixed<std::underlying_type_t<E1>>, "fixed");
enum class E2 {};
static_assert(!is_fixed<std::underlying_type_t<E2>>, "not fixed");
变量模板确实是从 C++14 开始的,但在 C++11 中,可以使用constexpr
函数或struct
/class
来实现相同的效果:
template <typename T>
constexpr bool is_fixed_f() {
return std::is_same<T, std::int8_t>::value ||
std::is_same<T, std::int16_t>::value
// etc..
;
}
template <typename T>
struct is_fixed_s {
static constexpr bool value =
std::is_same<T, std::int8_t>::value ||
std::is_same<T, std::int16_t>::value
// etc..
;
};
在回答标题问题时:不,不可能知道枚举是否具有显式底层类型。
即使有,它也不能解决你的实际问题,这更像是"如何知道枚举类型是否具有固定大小?
想象一下这个简单的案例:
enum class Foo : long {};
在某些系统上,这将是 32 位,而在另一些系统上,这将是 64 位。 因此,即使某些机制让您发现它具有显式类型,它也无济于事,因为大小不可移植。
- SFINAE是否取决于类型推断?
- 是否可以确定枚举是否为强类型?
- 在编译时将强类型枚举器转换为其基础类型?
- 如何检查联合是否包含类型(使用 type_traits)?
- 为什么不调用预期的函数?我是否对类型特征的理解不正确?
- 是否存在 x + 1 == x 类型的浮点型值 x
- 如果 int 是"not within the enums range",为什么将 int 转换为强类型枚举会编译?
- 使用 SFINAE 检查函数 std::to_string 是否存在类型
- Python C API:如何检查对象是否是类型的实例
- 原始类型的强类型(BOOST_STRONG_TYPEDEF没有切割)
- 检查是否完成类型
- 是否有类型特征显示一种类型是否可能包含其他类型的值
- 如何检测编译时是否shared_ptr类型
- 是否有类型特征检查类型之间的包含
- 强类型枚举的语法实现错误
- 警告 639:二进制操作中类型 'uint64' 的强类型不匹配
- 警告 634:相等或有条件的强类型不匹配(类型"bool")
- 枚举与强类型枚举
- 如何判断"->"运算符最终是否返回类型?
- 将基础类型的任意值强制转换为强类型枚举类型是否安全?