1 背景
在使用C语言开发中,使用printf打印64位变量比较常用,通常在32位系统(long long为64位)中使用%lld输出64位的变量,而在64位系统(long为64位)中则使用 %ld。
2 问题
如何解决跨平台问题呢?
代码如下:
#include <stdio.h>
#include <stdint.h>
int main(int argc, char *argv[])
{
int64_t a = 4294967295;
printf("a=%ld\n", a);
return 0;
}
- 64位系统运行结果:
a=4294967295
- 32位系统运行结果:
a=-1097716252
同样代码在32位系统上运行不正确。
修改代码:
#include <stdio.h>
#include <stdint.h>
int main(int argc, char *argv[])
{
int64_t a = 4294967295;
printf("a=%lld\n", a);
return 0;
}
在32位系统上运行正确,在64位系统上编译有如下警告:
print.c:6:18: warning: format ‘%lld’ expects argument of type ‘long long int’, but argument 2 has type ‘int64_t’ {aka ‘long int’} [-Wformat=]
6 | printf("a=%lld\n", a);
| ~~~^ ~
| | |
| | int64_t {aka long int}
| long long int
| %ld
有没有一套代码在32和64位系统上都正确运行呢?
3 解决
修改代码如下:
#include <stdio.h>
#include <stdint.h>
#include <inttypes.h> // for PRId64 PRIu64 PRIo64 PRIo64
int main(int argc, char *argv[])
{
int64_t a = 4294967295;
uint64_t b = 0xFFFFFFFFFFFF;
printf("a=%"PRId64"\n", a);//64位有符号10进制打印
printf("b=%"PRIu64"\n", b);//64位无符号10进制打印
printf("a=%"PRIo64"\n", a);//64位有符号8进制打印
printf("b=%"PRIo64"\n", b);//64位无符号8进制打印
return 0;
}
32位和64位运行结果:
a=4294967295
b=281474976710655
a=37777777777
b=7777777777777777
4 总结
C语言提供了标准头文件inttypes.h。在头文件中定义了这些宏PRId64,PRIu64,PRIo64,PRIo64,解决了64位数跨平台打印问题。