本文介绍了(为什么)使用未初始化的变量未定义的行为?的处理方法,对大家解决问题具有一定的参考价值,需要的朋友们下面随着小编来一起学习吧!

问题描述

如果我有:

unsigned int x;
x -= x;

很明显 x 应该 在这个表达式之后为零,但是我到处看,他们说这个代码的 behavior 是未定义的,不仅仅是 x 的值(直到减法之前).

it's clear that x should be zero after this expression, but everywhere I look, they say the behavior of this code is undefined, not merely the value of x (until before the subtraction).

两个问题:

  • 这段代码的行为是否确实未定义?
    (例如,代码可能会在兼容的系统上崩溃[或更糟]?)

  • Is the behavior of this code indeed undefined?
    (E.g. Might the code crash [or worse] on a compliant system?)

如果是这样,为什么 C 会说 行为 是未定义的,而 x 非常清楚 这里应该为零?

If so, why does C say that the behavior is undefined, when it is perfectly clear that x should be zero here?

即不在这里定义行为有什么优势?

i.e. What is the advantage given by not defining the behavior here?

显然,编译器可以简单地在变量中使用它认为方便"的 whatever 垃圾值,它会按预期工作......这种方法有什么问题?

Clearly, the compiler could simply use whatever garbage value it deemed "handy" inside the variable, and it would work as intended... what's wrong with that approach?

推荐答案

是的,这种行为是未定义的,但原因与大多数人所知道的不同.

Yes this behavior is undefined but for different reasons than most people are aware of.

首先,使用未定义的值本身并不是未定义的行为,但该值只是不确定的.如果该值恰好是该类型的陷阱表示,那么访问它就是 UB.无符号类型很少有陷阱表示,所以在这方面你会相对安全.

First, using an unitialized value is by itself not undefined behavior, but the value is simply indeterminate. Accessing this then is UB if the value happens to be a trap representation for the type. Unsigned types rarely have trap representations, so you would be relatively safe on that side.

使行为未定义的是变量的附加属性,即它可以用 register 声明",即它的地址永远不会被占用.此类变量被特殊处理,因为有些架构具有真正的 CPU 寄存器,这些寄存器具有一种未初始化"的额外状态,并且与类型域中的值不对应.

What makes the behavior undefined is an additional property of your variable, namely that it "could have been declared with register" that is its address is never taken. Such variables are treated specially because there are architectures that have real CPU registers that have a sort of extra state that is "uninitialized" and that doesn't correspond to a value in the type domain.

标准的相关短语是6.3.2.1p2:

The relevant phrase of the standard is 6.3.2.1p2:

如果左值指定一个自动存储持续时间的对象,可以用寄存器存储类声明(从来没有它的地址被占用),并且该对象未初始化(未声明使用初始化程序并且之前没有对其进行分配使用),行为未定义.

为了清楚起见,以下代码在所有情况下都是合法的:

And to make it clearer, the following code is legal under all circumstances:

unsigned char a, b;
memcpy(&a, &b, 1);
a -= a;

  • 这里取的是ab的地址,所以它们的值就是不确定.
  • 因为 unsigned char 从来没有陷阱表示该不确定值只是未指定,任何 unsigned char 值都可以发生.
  • 最后a 必须保持值0.
    • Here the addresses of a and b are taken, so their value is justindeterminate.
    • Since unsigned char never has trap representationsthat indeterminate value is just unspecified, any value of unsigned char couldhappen.
    • At the end a must hold the value 0.
    • Edit2: ab 有未指定的值:

      a and b have unspecified values:

      3.19.3 未指定值
      相关类型的有效值,其中本国际标准对哪个值没有要求在任何情况下都被选中

      这篇关于(为什么)使用未初始化的变量未定义的行为?的文章就介绍到这了,希望我们推荐的答案对大家有所帮助,也希望大家多多支持!

08-21 13:33