- android - RelativeLayout 背景可绘制重叠内容
- android - 如何链接 cpufeatures lib 以获取 native android 库?
- java - OnItemClickListener 不起作用,但 OnLongItemClickListener 在自定义 ListView 中起作用
- java - Android 文件转字符串
给定 enum class val { foo = 1, bar = 2, baz = 4 };
可以定义:
val operator|(val x, val y)
{
return static_cast<val>(static_cast<int>(x) | static_cast<int>(y));
}
但是,这样做在语义上是否正确?
我倾向于不,如下面看似良好的示例所示:
int convert(val x)
{
switch(x)
{
case val::foo: return 42;
case val::bar: return 53;
case val::baz: return 64;
}
}
调用 convert(val::foo | val::bar)
将在使用 g++ 编译和使用 clang++ 进行段错误时返回 0
。
我的问题有两个方面:
1.a 在上面的链接示例中哪个编译器是正确的,g++ 还是 clang++?
我可以想到几种可能的实现方式:
enum class val { foo, bar, baz, size };
using val_flags = std::set<val>; // (1)
using val_flags = std::vector<bool>; // (2)
using val_flags = std::bitset<val::size>; // (3)
using val_flags = std::underlying_type<val>::type; // (4)
更新:
谢谢大家的回答。我最终复活了我的旧枚举运算符模板。如果有人感兴趣,可以在这里找到:github.com
最佳答案
following, seemingly well-behaving example:
不是,但做一个小改动:
int convert(val x)
{
switch(x)
{
case val::foo: return 42;
case val::bar: return 53;
case val::baz: return 64;
}
return 9; // ADDED THIS LINE
}
一切都会好起来的。另一种解决方法是使用 default:
案例并返回那里。
您的现有代码通过到达具有非void
返回类型的函数的右大括号触发未定义的行为1。因为它是未定义的行为,所以两个编译器都是正确的。
在 enum
类型中保存值的语义是明确定义和保证的,这些值是枚举值的按位或组合。该标准要求 enum
的实例可以存储任何整数值,使用的位数不超过定义的任何枚举器值,其中包括所有按位或组合。以前的正式语言说这有点乱,但在这里(注意你的情况是一个 enum class
,这些总是有固定的底层类型并且第一句适用):
For an enumeration whose underlying type is fixed, the values of the enumeration are the values of the underlying type. Otherwise, for an enumeration where emin is the smallest enumerator and emax is the largest, the values of the enumeration are the values in the range bmin to bmax, defined as follows: Let K be 1 for a two’s complement representation and 0 for a ones’ complement or sign-magnitude representation. bmax is the smallest value greater than or equal to max(|emin| − K, |emin|) and equal to 2M − 1, where M is a non-negative integer. bmin is zero if emin is non-negative and −(bmax + K) otherwise. The size of the smallest bit-field large enough to hold all the values of the enumeration type is max(M, 1) if bmin is zero and M + 1 otherwise. It is possible to define an enumeration that has values not defined by any of its enumerators. If the enumerator-list is empty, the values of the enumeration are as if the enumeration had a single enumerator with value 0.
(来自 n4582,第 7.2 节 [dcl.enum]
)
1 从 6.6.3 [stmt.return]
:
Flowing off the end of a constructor, a destructor, or a function with a cv
void
return type is equivalent to a return with no operand. Otherwise, flowing off the end of a function other than main (3.6.1) results in undefined behavior.
关于c++ - 强制转换为枚举无法表示的枚举值是否合法?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/39455328/
我是一名优秀的程序员,十分优秀!