在考虑这个问题的反例时,我想出了:
struct A
{
alignas(2) char byte;
};
但是,如果这是合法且标准的布局,那么它的布局是否与 this 兼容struct B
?
struct B
{
char byte;
};
此外,如果我们有
struct A
{
alignas(2) char x;
alignas(4) char y;
};
// possible alignment, - is padding
// 00 01 02 03 04 05 06 07 08 09 10 11 12 13 14 15
// x - - - y - - - x - - - y - - -
struct B
{
char x;
char y;
}; // no padding required
union U
{
A a;
B b;
} u;
A
和有一个共同的初始序列B
吗?如果是这样,它是否包括A::y
& B::y
?即,我们可以编写以下不调用 UB 的代码吗?
u.a.y = 42;
std::cout << u.b.y;
(也欢迎 C++1y/“固定 C++11”的答案)
有关对齐,请参见 [basic.align],对于对齐说明符,请参见 [dcl.align]。
[basic.types]/11 表示基本类型“如果两个类型
T1
和T2
是相同类型,那么T1
和T2
是布局兼容类型。” (一个潜在的问题是是否A::byte
并且B::byte
具有布局兼容的类型)[class.mem]/16 “如果两个标准布局结构类型具有相同数量的非静态数据成员并且相应的非静态数据成员(按声明顺序)具有布局兼容类型,则它们是布局兼容的。”
[class.mem]/18 “如果对应的成员具有布局兼容的类型,并且两个成员都不是位域或两者都是具有相同宽度的位域,则两个标准布局结构共享一个共同的初始序列。或更多初始成员。”
[class.mem]/18 “如果一个标准布局联合包含两个或多个共享一个共同初始序列的标准布局结构,并且如果标准布局联合对象当前包含这些标准布局结构之一,则允许检查其中任何一个的共同初始部分。”
当然,在语言律师的层面上,另一个问题是“允许”对公共初始序列的检查意味着什么。我猜其他一些段落可能会导致上述u.b.x
未定义的行为(从未初始化的对象中读取)。