在Solaris 9上使用printf + uint_64?

时间:2020-03-06 15:03:26  来源:igfitidea点击:

我有一些使用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"

我可以使用什么?

解决方案

我们可以长时间使用%llu。但是,这也不是很容易移植,因为不能保证long long是64位。 :-)

如果我们有可用的inttypes.h,则可以使用它提供的宏:

printf(  "%" PRIu64 "\n", val);

不太漂亮(我最近似乎在说很多),但是可以。

如果要避免SunOS条件typedef,可以从stdint.h中获取uint64_t。

答案取决于代码实际上是C还是C ++。在C语言中,我们应该使用unsigned long long而不是其他类型(这与当前标准一致,并且就C99支持而言,long long很常见),在其后添加了ULL而不是L将其设为常量,并使用(如前所述)%llu作为说明符。如果不存在对C99的支持,则可能需要检查编译器文档,因为没有其他标准方法可以执行此操作。保证" long long"至少为64位。

在符合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。