众所周知,有符号整数溢出是未定义的行为.但是C++ 11 cstdint
文档中有一些有趣的东西:
有符号整数类型,宽度分别为8,16,32和64位,没有填充位,负值使用2的补码(仅当实现直接支持该类型时提供)
这里是我的问题:由于标准明确地说,对int8_t
,int16_t
,int32_t
和int64_t
负数是2的补,仍然是这些类型的未定义行为的泛滥?
编辑我检查了C++ 11和C11标准,这是我发现的:
C++ 11,§18.4.1:
标题定义了所有函数,类型和宏,与C标准中的7.20相同.
C11,§7.20.1.1:
typedef名称
intN_t
指定有符号整数类型,其宽度为N,无填充位和二进制补码表示.因此,int8_t
表示这样的带符号整数类型,其宽度恰好为8位.
通过这个回答我才知道:
由于计算而导致的有符号溢出在 C++20 中仍然是未定义的行为,而由于转换而导致的有符号溢出在 C++20 中得到了明确定义(这是为 Pre-C++20定义的实现)。
由于转换导致的有符号溢出的变化是因为 C++20 编译器要求使用 2 的补码。
我的问题是:
如果编译器需要使用 C++20 中的 2 的补码,那么为什么由于计算而导致的有符号溢出不像由于转换而导致的有符号溢出那样被明确定义?
也就是说,计算溢出和转换溢出之间为什么(如何)存在差异。从本质上讲,为什么这两种溢出的处理方式不同。
具体参考:https://blogs.msdn.microsoft.com/oldnewthing/20140627-00/?p = 633 /
我是一名新的C++程序员,我目前正在学习未定义的行为及其对程序的影响.我特意链接到上面的博客,该博客说当发生未定义的行为时,任何事情都可能发生.
它特别提到了几次,当发生未定义的行为时,编译器可以允许任何事情发生.
具体是什么导致这种情况发生,为什么会发生?
我有一个遗留的代码库,我们正试图从中迁移devtoolset-4
到devtoolset-7
.我注意到有关有符号整数溢出的有趣行为(int64_t
具体而言).
有一个代码片段,用于检测整数溢出,同时乘以一大组整数:
// a and b are int64_t
int64_t product = a * b;
if (b != 0 && product / b != a) {
// Overflow
}
Run Code Online (Sandbox Code Playgroud)
这段代码与devtoolset-4一起工作正常.但是,使用devtoolset-7时,永远不会检测到溢出.
例如:当a = 83802282034166
和b = 98765432
,
product
变为-5819501405344925872
(显然值已溢出).
但product / b
结果价值等于a (83802282034166)
.因此,if
情况永远不会成真.它的值应该根据overflown(负)product
值计算:-5819501405344925872 / 98765432 = -58922451788
具有讽刺意味的是,数学是正确的,但它导致devtoolset-4的异常行为.
product / b != a
为product != a * b
并达到相同的溢出值(或者可能只是根据上面的语句跳过计算 …