一尘不染

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

linux

在我的程序中,我统计了他们想要的文件并发送了数据。统计信息的字段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位系统上运行。现在,使用转换似乎可以正常工作,但是可以在几个平台上使用?


阅读 940

收藏
2020-06-02

共1个答案

一尘不染

没有一种完全可移植的方式来执行此操作,这很麻烦。

C99提供了一种针对内置类型的机制,例如size_t带有%zu符号(并且还有一些额外的类似的限定词)。

它还为<inttypes.h>标头提供诸如PRIX32之类的宏,以定义用于打印32位十六进制常数的正确限定符(在这种情况下):

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

对于系统定义的类型(例如POSIX定义的类型),AFAIK,没有好的方法来处理它们。因此,我要做的是快速猜测“安全”转换,然后据此进行打印,包括演员表,这就是您在问题中所说明的。这令人沮丧,但是我知道没有更好的方法。如有疑问,请使用C99,然后转换为“unsignedlonglong”是相当不错的;可能需要使用uintmax_t强制转换为PRIXMAX或等效版本。

或者,正如FUZxxl提我的那样,您可以使用修饰符j指示“最大”整数类型。例如:

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