2009-09-09 2 views
26

В моей программе я устанавливаю файлы, которые они хотят, и отправляет данные. Поля стата struct все специальные типы:Как использовать printf для отображения off_t, nlink_t, size_t и других специальных типов?

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-битной системе. Теперь с кастами это работает, но на сколько платформ?

+0

Специфический для clock_t: http://stackoverflow.com/questions/1083142/what-s-the-correct-way-to-use-printf-to-print-a-clock-t –

+0

Для '__uint128': http://stackoverflow.com/questions/11656241/how-to-print-uint128-t-number-using-gcc –

ответ

25

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

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 или эквивалент.

Или, как FUZxxlreminded me, вы можете использовать модификатор j, чтобы указать целочисленный тип 'max'. Например:

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

Не мог бы сказать это лучше. Использование (u) intmax_t и PRI (d | u) MAX (или эквивалент) может быть лучшим решением для большинства проблемных системных типов. –

+1

Я уже знал эти макросы, но они казались такими уродливыми для меня. Спасибо за совет. – kmm

+2

Они уродливые, но это ближайшие вещи, которые я знаю. –