我有一个size_t类型的变量,我想使用printf()打印它。我使用什么格式说明符来可移植地打印它?
在32位机器中,%u似乎是正确的。我用g++ -g -W -Wall -Werror -ansi -pedantic编译,没有警告。但是当我在64位机器上编译该代码时,它会产生警告。
size_t x = <something>;
printf("size = %u\n", x);
warning: format '%u' expects type 'unsigned int',
but argument 2 has type 'long unsigned int'
如预期的那样,如果我将其更改为%lu,警告就会消失。
问题是,我如何编写代码,使它在32位和64位机器上编译警告免费?
编辑:作为一种变通方法,我猜一个答案可能是将变量“强制转换”为一个足够大的整数,比如unsigned long,然后使用%lu打印。这在两种情况下都适用。我看看有没有其他的办法。
如果你想打印size_t的值作为字符串,你可以这样做:
char text[] = "Lets go fishing in stead of sitting on our but !!";
size_t line = 2337200120702199116;
/* on windows I64x or I64d others %lld or %llx if it works %zd or %zx */
printf("number: %I64d\n",*(size_t*)&text);
printf("text: %s\n",*(char(*)[])&line);
结果是:
号码:2337200120702199116
我们去钓鱼吧,别坐在椅子上了!
编辑:重新阅读问题,因为向下投票,我注意到他的问题不是%llu或%I64d,而是size_t类型在不同的机器上看到这个问题https://stackoverflow.com/a/918909/1755797
http://www.cplusplus.com/reference/cstdio/printf/
Size_t在32位机器上是unsigned long long int,在64位机器上是unsigned long long int
但是%ll总是期望一个unsigned long long int。
Size_t在不同的操作系统上的长度不同,而%llu是相同的
扩展亚当·罗森菲尔德对Windows的回答。
我在VS2013 Update 4和VS2015预览版上测试了这段代码:
// test.c
#include <stdio.h>
#include <BaseTsd.h> // see the note below
int main()
{
size_t x = 1;
SSIZE_T y = 2;
printf("%zu\n", x); // prints as unsigned decimal
printf("%zx\n", x); // prints as hex
printf("%zd\n", y); // prints as signed decimal
return 0;
}
VS2015生成二进制输出:
1
1
2
而VS2013生成的则是:
去
zx
zd
注意:ssize_t是POSIX扩展,ssize_t在Windows数据类型中是类似的东西,因此我添加了<BaseTsd.h>引用。
此外,除了以下C99/C11头文件,所有C99头文件都可以在VS2015预览版中使用:
C11 - <stdalign.h>
C11 - <stdatomic.h>
C11 - <stdnoreturn.h>
C99 - <tgmath.h>
C11 - <threads.h>
此外,C11的<uchar.h>现在包含在最新预览中。
有关更多详细信息,请参阅此旧列表和新列表以了解标准一致性。