我有一个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打印。这在两种情况下都适用。我看看有没有其他的办法。


当前回答

C99为此定义了“%zd”等。c++中没有可移植的格式说明符——你可以使用%p,这在这两种情况下都是word,但也不是一个可移植的选择,并以十六进制给出值。

或者,使用一些流(例如stringstream)或安全的printf替代品,如Boost Format。我知道这个建议只有有限的用途(并且确实需要c++)。(在实现unicode支持时,我们使用了适合我们需求的类似方法。)

C语言的基本问题是printf使用省省号在设计上是不安全的——它需要从已知参数中确定额外参数的大小,所以它不能被修复为支持“你得到的任何东西”。所以除非你的编译器实现了一些专有的扩展,否则你就不走运了。

其他回答

printf("size = %zu\n", sizeof(thing) );

如果您传递一个32位无符号整数为%lu格式,它会警告您吗?这应该没问题,因为转换是定义良好的,不会丢失任何信息。

我听说有些平台在<inttypes.h>中定义了宏,可以插入到格式字符串文字中,但我在Windows c++编译器上没有看到这个头,这意味着它可能不是跨平台的。

在某些平台上,对于某些类型,有特定的printf转换说明符可用,但有时必须求助于更大的类型强制转换。

我在这里用示例代码记录了这个棘手的问题: http://www.pixelbeat.org/programming/gcc/int_types/ 并定期更新有关新平台和类型的信息。

扩展亚当·罗森菲尔德对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>现在包含在最新预览中。

有关更多详细信息,请参阅此旧列表和新列表以了解标准一致性。

std::size_t s = 1024;
std::cout << s; // or any other kind of stream like stringstream!