如何使用printf显示off_t,nlink_t,size_t和其他特殊类型?

时间:2009-09-09 19:15:20

标签: c linux printf portability

在我的程序中,我统计他们想要的文件并发送数据。统计struct的字段都是特殊类型:

struct stat {
  dev_t     st_dev;     /* ID of device containing file */
  ino_t     st_ino;     /* inode number */
  mode_t    st_mode;    /* protection */
  nlink_t   st_nlink;   /* number of hard links */
  uid_t     st_uid;     /* user ID of owner */
  gid_t     st_gid;     /* group ID of owner */
  dev_t     st_rdev;    /* device ID (if special file) */
  off_t     st_size;    /* total size, in bytes */
  blksize_t st_blksize; /* blocksize for file system I/O */
  blkcnt_t  st_blocks;  /* number of 512B blocks allocated */
  time_t    st_atime;   /* time of last access */
  time_t    st_mtime;   /* time of last modification */
  time_t    st_ctime;   /* time of last status change */
};

我的问题的相关代码如下:

len = snprintf( statbuf, STAT_BUFFER_SIZE,
  "%crwxrwxrwx %lu %u %u %lld %s %s\r\n",
  S_ISDIR( filestats.st_mode ) ? 'd' : '-',
  (unsigned long ) filestats.st_nlink,
  filestats.st_uid,
  filestats.st_gid,
  (unsigned long long ) filestats.st_size,
  date,
  filename);

如何以便携和有效的方式打印这些类型?起初我通过猜测正确的格式说明符而没有强制转换。除了令人讨厌的编程习惯外,这也意味着我的代码无法在32位系统上运行。现在有了演员阵容似乎有效,但有多少平台?

1 个答案:

答案 0 :(得分:26)

没有一种完全可移植的方法,这是一种麻烦。

C99为size_t等内置类型提供了%zu符号的机制(还有一些额外的类似限定符)。

它还为<inttypes.h>标头提供了诸如PRIX32之类的宏,以定义用于打印32位十六进制常量的正确限定符(在本例中):

printf("32-bit integer: 0x%08" PRIX32 "\n", var_of_type_int32_t);

对于系统定义的类型(例如POSIX定义的类型),AFAIK,没有好办法处理它们。所以,我所做的就是对“安全”转换进行猜测,然后进行相应的打印,包括演员,这就是你在问题中所说明的内容。这令人沮丧,但我知道没有更好的方法。如果有疑问,并使用C99,那么转换为'unsigned long long'是相当不错的;可能存在使用强制转换为uintmax_t和PRIXMAX或等效的情况。

或者,作为FUZxxl reminded我,您可以使用修饰符j来表示“最大”整数类型。例如:

printf("Maximal integer: 0x%08jX\n", (uintmax_t)var_of_type_without_format_letter);