2009-09-09 4 views
26

W moim programie wybieram żądane pliki i przesyłam dane. Pola stat struct są specjalne wszystkie rodzaje:Jak korzystać z printf do wyświetlania off_t, nlink_t, size_t i innych typów specjalnych?

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 */ 
}; 

Odpowiedni kod na moje pytanie w następujący sposób:

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); 

Jak wydrukować te typy w przenośnym i skuteczny sposób? Na początku zrobiłem to bez rzutów, odgadując poprawne specyfikatory formatu. Oprócz bycia irytującym nawykiem programowania, oznaczało to również, że mój kod nie działałby na systemie 32-bitowym. Teraz z obsadami wydaje się działać, ale na ilu platformach?

+0

Specyficzne dla clock_t: http://stackoverflow.com/questions/1083142/what-s-the-correct-way-to-use-printf-to-print-a-clock-t –

+0

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

Odpowiedz

25

Nie ma w pełni przenośnego sposobu na zrobienie tego i jest to uciążliwe.

C99 udostępnia mechanizm wbudowanych typów, takich jak size_t z notacją %zu (i jest kilka dodatkowych, podobnych kwalifikacji).

zawiera również nagłówek makr jak PRIX32 <inttypes.h> do określenia prawidłowej kwalifikator do drukowania 32-bitową stałą szesnastkowe (w tym przypadku):

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

odniesieniu do typów systemów zdefiniowanych (np określone przez POSIX), AFAIK, nie ma dobrego sposobu na ich obsługę. Tak więc, robię latające domysły na "bezpieczną" konwersję, a następnie drukuję odpowiednio, w tym obsadę, co ilustrujesz w pytaniu. To frustrujące, ale nie ma lepszego sposobu, który znam. W razie wątpliwości i przy użyciu C99, konwersja na "unsigned long long" jest całkiem dobra; może istnieć przypadek użycia rzutowania do uintmax_t i PRIXMAX lub odpowiednika.

Lub, jako FUZxxlreminded mnie, możesz użyć modyfikatora j, aby wskazać "maks." Typ całkowity. Na przykład:

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

Nie mogłem powiedzieć tego lepiej. Używanie (u) intmax_t i PRI (d | u) MAX (lub równoważny) może być najlepszym rozwiązaniem dla większości problematycznych typów zdefiniowanych przez system. –

+1

Znałem już te makro, ale dla mnie wyglądały tak brzydko. Dzięki za radę. – kmm

+2

Są brzydkie, ale są najbliżej przenośne, które znam. –