Программирование UNIX. структура timeval, как распечатать его (C-программирование)
Я пытаюсь напечатать значение типа timeval. На самом деле я могу распечатать его, но получаю следующее предупреждение:
Несколько маркеров на этой линии
- формат "%ld" ожидает тип "long int", но аргумент 2 имеет тип "struct timeval"
Программа компилируется и печатает значения, но я хотел бы знать, что я делаю что-то не так. Благодарю.
printf("%ld.%6ld\n",usage.ru_stime);
printf("%ld.%6ld\n",usage.ru_utime);
где использование имеет тип
typedef struct{
struct timeval ru_utime; /* user time used */
struct timeval ru_stime; /* system time used */
long ru_maxrss; /* maximum resident set size */
long ru_ixrss; /* integral shared memory size */
long ru_idrss; /* integral unshared data size */
long ru_isrss; /* integral unshared stack size */
long ru_minflt; /* page reclaims */
long ru_majflt; /* page faults */
long ru_nswap; /* swaps */
long ru_inblock; /* block input operations */
long ru_oublock; /* block output operations */
long ru_msgsnd; /* messages sent */
long ru_msgrcv; /* messages received */
long ru_nsignals; /* signals received */
long ru_nvcsw; /* voluntary context switches */
long ru_nivcsw; /* involuntary context switches */
}rusage;
struct rusage usage;
5 ответов
В библиотеке GNU Cstruct timeval
:
объявлен в sys/time.h и имеет следующие члены:
long int tv_sec
Это представляет количество полных секунд прошедшего времени.
long int tv_usec
Это оставшееся время (доля секунды), представленное в виде количества микросекунд. Это всегда меньше миллиона.
Так что вам нужно будет сделать
printf("%ld.%06ld\n", usage.ru_stime.tv_sec, usage.ru_stime.tv_usec);
чтобы получить "красиво отформатированную" временную метку, например 1.000123
,
Поскольку struct timeval
будет объявлено что-то вроде:
struct timeval {
time_t tv_sec;
suseconds_t tv_usec;
}
вам нужно добраться до базовых полей:
printf ("%ld.%06ld\n", usage.ru_stime.tv_sec, usage.ru_stime.tv_usec);
printf ("%ld.%06ld\n", usage.ru_utime.tv_sec, usage.ru_utime.tv_usec);
Да его
int main( void )
{
clock_t start, stop;
long int x;
double duration;
static struct timeval prev;
struct timeval now;
start = clock(); // get number of ticks before loop
for( x = 0; x < 1000000000; x++ );
// sleep(100);
stop = clock(); // get number of ticks after loop
// calculate time taken for loop
duration = ( double ) ( stop - start ) / CLOCKS_PER_SEC;
printf( "\nThe number of seconds for loop to run was %.2lf\n", duration );
gettimeofday(&now, NULL);
prev.tv_sec = duration;
if (prev.tv_sec)
{
int diff = (now.tv_sec-prev.tv_sec)*1000+(now.tv_usec-prev.tv_usec)/1000;
printf("DIFF %d\n",diff);
}
return 0;
}
Я только что сделал эту удобную маленькую функцию из информации выше. Самостоятельный, за исключением необходимости времени. Назовите его с нужной вам меткой, где бы вы не хотели знать время в потоке стандартного вывода.
void timestamp(char *lbl) { // just outputs time and label
struct timeval tval;
int rslt;
rslt = gettimeofday(&tval,NULL);
if (rslt) printf("gettimeofday error\n");
printf("%s timestamp: %ld.%06ld\n", lbl, tval.tv_sec, tval.tv_usec);
}
Типичный вывод выглядит так: dpyfunc got ftqmut timestamp: 1537394319.501560
И вы можете окружить вызовы к нему #ifdef, чтобы включать и выключать их все сразу, комментируя ваш #define. Это может быть полезно почти как профилирование, но вы можете быстро отключить его для кода производства / выпуска. Подобно:
#define TIMEDUMPS
#ifdef TIMEDUMPS
timestamp("function 1 start");
#endif
#ifdef TIMEDUMPS
timestamp("function 2 start");
#endif
Закомментируйте #define TIMEDUMPS, и он отключит их все. Неважно, сколько, во сколько файлов исходного кода.
.tv_sec может быть -ve, и когда это происходит, .tv_usec смещается (принудительно в диапазоне [0..1000000), следовательно:
#include <stdint.h>
#include <stdlib.h>
#include <stdio.h>
#include <sys/time.h>
static void frac(intmax_t usec)
{
int precision = 6;
while (usec % 10 == 0 && precision > 1) {
precision--;
usec = usec / 10;
}
printf(".%0*jd", precision, usec);
}
static void print_timeval(struct timeval tv)
{
struct timeval d;
/*
* When .tv_sec is -ve, .tv_usec is biased (it's forced to the
* range 0..1000000 and then .tv_sec gets adjusted). Rather
* than deal with that convert -ve values to +ve.
*/
if (tv.tv_sec < 0) {
printf("-");
struct timeval zero = {0,};
timersub(&zero, &tv, &d);
} else {
d = tv;
}
printf("%jd", (intmax_t)d.tv_sec);
if (d.tv_usec > 0) {
frac(d.tv_usec);
}
}
int main()
{
for (intmax_t i = 1000000; i > 0; i = i / 10) {
for (intmax_t j = 1000000; j > 0; j = j / 10) {
struct timeval a = { .tv_sec = i / 1000000, .tv_usec = i % 1000000, };
struct timeval b = { .tv_sec = j / 1000000, .tv_usec = j % 1000000, };
struct timeval d;
timersub(&a, &b, &d);
printf("%7jd us - %7jd us = %7jd us | %2jd.%06jd - %2jd.%06jd = %2jd.%06jd | ",
i, j, i - j,
(intmax_t)a.tv_sec, (intmax_t)a.tv_usec,
(intmax_t)b.tv_sec, (intmax_t)b.tv_usec,
(intmax_t)d.tv_sec, (intmax_t)d.tv_usec);
print_timeval(d);
printf("\n");
}
}
return 0;
}
Да, timeval определяется так
struct timeval {
time_t tv_sec;
suseconds_t tv_usec;
}
С помощью
printf ("%ld.%06ld\n", usage.ru_stime.tv_sec, usage.ru_stime.tv_usec);
несомненно, поможет.