题
我有一些使用sprintf将uint_64转换为字符串的c(++)代码。这需要可移植到Linux和Solaris。
在Linux上我们使用%ju,但在Solaris上似乎没有任何等价物。我能找到的最接近的是%lu,但这会产生不正确的输出。一些示例代码:
#include <stdio.h>
#include <sys/types.h>
#ifdef SunOS
typedef uint64_t u_int64_t;
#endif
int main(int argc, char **argv) {
u_int64_t val = 123456789123L;
#ifdef SunOS
printf("%lu\n", val);
#else
printf("%ju\n", val);
#endif
}
在linux上,输出符合预期;在Solaris 9上(不要问),它是“28”
我可以使用什么?
解决方案
如果你有inttypes.h,你可以使用它提供的宏:
printf( "%" PRIu64 "\n", val);
不漂亮(我最近似乎说了很多),但它确实有效。
其他提示
在符合C99的系统上:
#include <inttypes.h>
uint64_t big = ...;
printf("%" PRIu64 "\n", big);
见C99标准第7.8节。
说明符是{PRI,SCN} [diouxX] {N,LEASTN,MAX,FASTN,PTR}
PRI用于printf()系列,SCN用于scanf()系列,d和i用于有符号整数类型; o,u,x,X用于无符号整数类型,如八进制,十进制,十六进制和十六进制; N是支持的宽度之一; LEAST和FAST对应于那些修饰符; PTR用于intptr_t;和MAX用于intmax_t。
答案取决于您的代码实际上是C还是C ++。在C中,你应该使用 unsigned long long
而不是另一种类型(这符合当前标准,并且 long long
是非常常见的,只要C99支持如此),将 ULL
而不是 L
附加到常量,并使用(如前所述)%llu
作为您的说明符。如果不存在对C99的支持,您可能需要检查编译器文档,因为没有其他标准方法可以执行此操作。 long long
至少被保证为64位。
您可以长时间使用%llu
。但是,这也不是很便携,因为 long long
不能保证是64位。 : - )
如果要避免使用SunOS条件typedef,可以从 stdint.h
获取 uint64_t
。