Как использовать printf для отображения off_t, nlink_t, size_t и других специальных типов?

В моей программе я ставил файлы, которые они хотят, и отправлял данные. Поля stat 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

Существует не полностью переносимый способ сделать это, и это неприятно.

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 напомнил меня, вы можете использовать модификатор j для указания a 'max' целочисленный тип. Например:

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