我想在我的代码中定义千兆字节,所以我首先使用了unsigned long
。但是,unsigned long
无法处理2 * gigabyte
。
所以,我用long long
替换了它,但我得到了同样的编译错误/警告: 错误:表达式中的整数溢出 [-Werror=溢出]
最后,我查找了大整数,发现uintmax_t是我需要的,因为它是 128 位。
不幸的是,我仍然遇到同样的错误。我想有一个小故障,但我可以找到它。
请在下面找到相关代码:
#define kilobyte 1024
#define megabyte 1024 * kilobyte
#define gigabyte 1024 * megabyte
uintmax_t threshold = 2 * gigabyte;
最后,在运行"make"之后
g++ -Wall -Wextra -Werror -pedantic -pthread -std=c++0x -g -o lcr main.cpp
我得到了:
main.cpp: In function ‘int main(int, char**)’:
main.cpp:104:17: error: integer overflow in expression [-Werror=overflow]
cc1plus: all warnings being treated as errors
两个int
的乘积显然仍然是 int
型 - 我们在宏中处理的文字是 int
型。
10243 = 230 在 32 位int
中几乎可以表示。但是,2*230 = 231 对于 32 位有符号int
来说正好太大。
这可以通过将常量定义为适当类型的对象来解决:
const std::uintmax_t kilobyte = 1024;
const std::uintmax_t megabyte = 1024 * kilobyte;
const std::uintmax_t gigabyte = 1024 * megabyte;
const std::uintmax_t threshold = 2 * gigabyte;
由于通常的算术转换是在*
的操作数上执行的,因此不会发生溢出。
让我们看一段代码:
uintmax_t x = 2 * 1024;
这里发生的事情是,我们有(int) 2 * (int) 1024
,然后我们将结果提升到uintmax_t
。
相反,我们想要:(uintmax_t) 2 * (uintmax_t) 1024
.我们可以通过此方法轻松地将整数提升为long long
:
#define kilobyte 1024ULL
#define megabyte 1024ULL * kilobyte
#define gigabyte 1024ULL * megabyte
uintmax_t threshold = 2ULL * gigabyte;