- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
问题:如果 union 包含两个具有兼容类型的公共(public)初始序列的结构,那么如果我们使用一个结构初始化初始序列的某些部分并使用初始序列的其余部分,则行为定义明确使用另一个结构的序列?
考虑以下代码片段:
union u_t{
struct {
int i1;
int i2;
} s1;
struct {
int j1;
int j2;
} s2;
};
int main(){
union u_t *u_ptr = malloc(sizeof(*u_ptr));
u_ptr -> s1.i1 = 10;
u_ptr -> s2.j2 = 11;
printf("%d\n", u_ptr -> s2.j1 + u_ptr -> s1.i2); //prints 21
}
问题是“打印 21”行为是否定义明确。标准 N1570 6.5.2.3(p6)
规定了以下内容:
if a union contains several structures that share a common initial sequence (see below), and if the union object currently contains one of these structures, it is permitted to inspect the common initial part of any of them anywhere that a declaration of the completed type of the union is visible.
因此可以检查公共(public)初始序列(在本例中为整个结构)。但问题是,在这种情况下, union 似乎包含 s2
对象,其中 j2
是唯一的初始化成员。
我认为我们最终会出现未指定行为,因为我们只初始化了 s2.j2
而 s2.j1
做了不是,所以它应该包含未指定的值。
最佳答案
关于别名:
公共(public)初始序列只关心两种结构类型的别名。这在这里不是问题,您的两个结构甚至是兼容的类型,因此指向它们的指针可能会在不使用任何技巧的情况下产生别名。剖析 C11 6.2.7:
6.2.7 Compatible type and composite type
Two types have compatible type if their types are the same. /--/ Moreover, two structure, union, or enumerated types declared in separate translation units are compatible if their tags and members satisfy the following requirements:If one is declared with a tag, the other shall be declared with the same tag.
这两个结构都没有在此处使用标记声明。
If both are completed anywhere within their respective translation units, then the following additional requirements apply:
它们都已完成(已定义)。
there shall be a one-to-one correspondence between their members such that each pair of corresponding members are declared with compatible types;
这适用于这些结构。
if one member of the pair is declared with an alignment specifier, the other is declared with an equivalent alignment specifier; and if one member of the pair is declared with a name, the other is declared with the same name.
对齐说明符不适用。
For two structures, corresponding members shall be declared in the same order.
这是正确的。
结论是你的两个结构都是兼容的类型。这意味着您不需要像通用初始序列这样的任何技巧。严格的别名规则简单地说明 (6.5/7):
An object shall have its stored value accessed only by an lvalue expression that has one of the following types:
— a type compatible with the effective type of the object,
这里就是这种情况。
此外,如其他答案所述,此处实际数据的有效类型是int
,因为分配的存储不会产生有效类型,因此它成为第一个用于左值访问的类型。这也意味着编译器不能假设指针不会别名。
此外,严格的别名规则为结构和 union 成员的左值访问提供了一个异常(exception):
an aggregate or union type that includes one of the aforementioned types among its members
然后你在上面有一个共同的初始序列。就别名而言,这是尽可能明确的定义。
关于类型双关语:
您真正关心的似乎不是别名,而是通过 union 输入双关语。 C11 6.5.2.3/3 模糊地保证了这一点:
A postfix expression followed by the . operator and an identifier designates a member of a structure or union object. The value is that of the named member,95) and is an lvalue if the first expression is an lvalue.
那是规范文本,写得很糟糕——没有人能理解程序/编译器应该如何基于此运行。内容丰富的脚注 95) 解释得很好:
95) If the member used to read the contents of a union object is not the same as the member last used to store a value in the object, the appropriate part of the object representation of the value is reinterpreted as an object representation in the new type as described in 6.2.6 (a process sometimes called "type punning"). This might be a trap representation.
在您的情况下,您触发了从一种结构类型到另一种兼容结构类型的类型转换。这是非常安全的,因为它们是完全相同的类型,并且对齐或陷阱的问题不适用。
请注意这里的 C++ 是不同的。
关于c - 使用公共(public)初始序列初始化两个结构的并集,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/54705564/
我是 Spring 新手,这就是我想要做的事情: 我正在使用一个基于 Maven 的库,它有自己的 Spring 上下文和 Autowiring 字段。 它的bean配置文件是src/test/res
我在我的测试脚本中有以下列表初始化: newSequenceCore=["ls", "ns", "*", "cm", "*", "ov", "ov", "ov", "ov", "kd"] (代表要在控
这个问题在这里已经有了答案: 关闭 11 年前。 Possible Duplicate: Class construction with initial values 当我查看 http://en.
我得到了成员变量“objectCount”的限定错误。编译器还返回“ISO C++ 禁止非常量静态成员的类内初始化”。这是主类: #include #include "Tree.h" using n
我有如下所示的a.h class A { public: void doSomething()=0; }; 然后我有如下所示的b.h #include "a.h" class b: publi
我需要解析 Firebase DataSnapshot (一个 JSON 对象)转换成一个数据类,其属性包括 enum 和 list。所以我更喜欢通过传递 DataSnapshot 来手动解析它进入二
我使用 JQuery 一段时间了,我总是使用以下代码来初始化我的 javascript: $(document).ready( function() { // Initalisation logic
这里是 Objective-C 菜鸟。 为什么会这样: NSString *myString = [NSString alloc]; [myString initWithFormat:@"%f", s
我无法让核心数据支持的 NSArrayController 在我的代码中正常工作。下面是我的代码: pageArrayController = [[NSArrayController alloc] i
我对这一切都很陌生,并且无法将其安装到我的后端代码中。它去哪里?在我的页脚下面有我所有的 JS? 比如,这是什么意思: Popup initialization code should be exec
这可能是一个简单的问题,但是嘿,我是初学者。 所以我创建了一个程序来计算一些东西,它目前正在控制台中运行。我决定向其中添加一个用户界面,因此我使用 NetBeans IDE 中的内置功能创建了一个 J
我有 2 个 Controller ,TEST1Controller 和 TEST2Controller 在TEST2Controller中,我有一个initialize()函数设置属性值。 如果我尝
据我所知, dependentObservable 在声明时会进行计算。但如果某些值尚不存在怎么办? 例如: var viewModel ={}; var dependentObservable1 =
我正在阅读 POODR 这本书,它使用旧语法进行默认值初始化。我想用新语法实现相同的功能。 class Gear attr_reader :chainring, :cog, :wheel de
我按照 polymer 教程的说明进行操作: https://www.polymer-project.org/3.0/start/install-3-0 (我跳过了可选部分) 但是,在我执行命令“po
很抱歉问到一个非常新手的Kotlin问题,但是我正在努力理解与构造函数和初始化有关的一些东西。 我有这个类和构造函数: class TestCaseBuilder constructor(
假设我们有一个包含 30 列和 30 行的网格。 生命游戏规则简而言之: 一个小区有八个相邻小区 当一个细胞拥有三个存活的相邻细胞时,该细胞就会存活 如果一个细胞恰好有两个或三个活的相邻细胞,那么它就
我是 MQTT 和 Android 开放附件“AOA” 的新手。在阅读教程时,我意识到,在尝试写入 ByteArrayOutputStream 类型的变量之前,应该写入 0 或 0x00首先到该变量。
我有 2 个 Controller ,TEST1Controller 和 TEST2Controller 在TEST2Controller中,我有一个initialize()函数设置属性值。 如果我尝
我有一个inotify /内核问题。我正在使用“inotify” Python项目进行观察,但是,我的问题仍然是固有的关于inotify内核实现的核心。 Python inotify项目处理递归ino
我是一名优秀的程序员,十分优秀!