参考: http://www.cnblogs.com/krythur/archive/2013/02/25/2932647.html

第一章 获取时间函数

asctime

函数定义
char * asctime(const [struct](https://so.csdn.net/so/search?q=struct&spm=1001.2101.3001.7020) tm * timeptr);

函数说明
asctime() 将参数 timeptr 所指的 tm 结构中的信息转换成真实世界所使用的时间日期表示方法,然后将结果以字符串形态返回。此函数已经由时区转换成当地时间,字符串格式为:“Wed Jun 30 21:49:08 1993\n”

返回值
若再调用相关的时间日期函数,此字符串可能会被破坏。此函数与 ctime 不同处在于传入的参数是不同的结构。

附加说明
返回一字符串表示目前当地的时间日期。

示例

  1. #include <stdio.h>
  2. #include <time.h>
  3. int main(void)
  4. {
  5. time_t timep;
  6. time(&timep);
  7. printf("%s", asctime(gmtime(&timep)));
  8. return 0;
  9. }

运行结果
Tue Aug 2 11:44:14 2022

ctime

定义函数
char *ctime(const time_t *timep);

函数说明
ctime() 将参数 timep 所指的 time_t 结构中的信息转换成真实世界所使用的时间日期表示方法,然后将结果以字符串形态返回。此函数已经由时区转换成当地时间,字符串格式为 “Wed Jun 30 21 :49 :08 1993\n”。若再调用相关的时间日期函数,此字符串可能会被破坏。

返回值
返回一字符串表示目前当地的时间日期。

示例

  1. #include <stdio.h>
  2. #include <time.h>
  3. int main(void)
  4. {
  5. time_t timep;
  6. time(&timep);
  7. printf("%s", ctime(&timep));
  8. return 0;
  9. }

运行结果
Tue Aug 2 19:46:27 2022

gmtime

函数定义
struct tm* gmtime(const time_t*timep);

函数说明
gmtime() 将参数 timep 所指的 time_t 结构中的信息转换成真实世界所使用的时间日期表示方法,然后将结果由结构 tm 返回。

结构 tm 的定义为:

  1. struct tm {
  2. int tm_sec;
  3. int tm_min;
  4. int tm_hour;
  5. int tm_mday;
  6. int tm_mon;
  7. int tm_year;
  8. int tm_wday;
  9. int tm_yday;
  10. int tm_isdst;
  11. };

int tm_sec 代表目前秒数,正常范围为 0-59,但允许至 61 秒
int tm_min 代表目前分数,范围 0-59
int tm_hour 从午夜算起的时数,范围为 0-23
int tm_mday 目前月份的日数,范围 01-31
int tm_mon 代表目前月份,从一月算起,范围从 0-11
int tm_year 从 1900 年算起至今的年数
int tm_wday 一星期的日数,从星期一算起,范围为 0-6
int tm_yday 从今年 1 月 1 日算起至今的天数,范围为 0-365
int tm_isdst 日光节约时间的旗标
此函数返回的时间日期未经时区转换,而是 UTC 时间。

返回值
返回结构 tm 代表目前 UTC 时间

示例

  1. #include <stdio.h>
  2. #include <time.h>
  3. int main(void)
  4. {
  5. char *wday[] = { "Sun", "Mon", "Tue", "Wed", "Thu", "Fri", "Sat" };
  6. time_t timep;
  7. struct tm *p;
  8. time(&timep);
  9. p = gmtime(&timep);
  10. printf("%d %d %d\n", (1900 + p->tm_year), (1 + p->tm_mon), p->tm_mday);
  11. printf("%s %d:%d:%d\n", wday[p->tm_wday], p->tm_hour, p->tm_min, p->tm_sec);
  12. return 0;
  13. }

运行结果
2022 8 2
Tue 11:50:21

localtime

函数定义
struct tm *localtime(const time_t * timep);

函数说明
localtime() 将参数 timep 所指的 time_t 结构中的信息转换成真实世界所使用的时间日期表示方法,然后将结果由结构 tm 返回。结构 tm 的定义请参考 gmtime()。此函数返回的时间日期已经转换成当地时区。

返回值
返回结构 tm 代表目前的当地时间。

示例

  1. #include <stdio.h>
  2. #include <time.h>
  3. int main(void)
  4. {
  5. char *wday[] = { "Sun", "Mon", "Tue", "Wed", "Thu", "Fri", "Sat" };
  6. time_t timep;
  7. struct tm *p;
  8. time(&timep);
  9. p = localtime(&timep); /*取得当地时间*/
  10. printf("%d %d %d\n", (1900 + p->tm_year), 1 + p->tm_mon, p->tm_mday);
  11. printf("%s %d: %d: %d\n", wday[p->tm_wday], p->tm_hour, p->tm_min, p->tm_sec);
  12. return 0;
  13. }

运行结果
2022 8 2
Tue 19: 54: 50

mktime

函数定义
time_t mktime(strcut tm * timeptr);

函数说明
mktime() 用来将参数 timeptr 所指的 tm 结构数据转换成从公元 1970 年 1 月 1 日 0 时 0 分 0 秒算起至今的 UTC 时间所经过的秒数。

返回值
返回经过的秒数。

示例

  1. /* 用 time() 取得时间(秒数),利用 localtime()
  2. 转换成 struct tm 再利用 mktine()将 struct tm 转换成原来的秒数*/
  3. #include <stdio.h>
  4. #include <time.h>
  5. int main(void)
  6. {
  7. time_t timep;
  8. struct tm *p;
  9. time(&timep);
  10. printf("time() : %ld \n", timep);
  11. p = localtime(&timep);
  12. timep = mktime(p);
  13. printf("time()->localtime()->mktime() : %ld\n", timep);
  14. return 0;
  15. }

运行结果
time() : 1659441407
time()->localtime()->mktime() : 1659441407``

time

函数定义
time_t time(time_t *t);

函数说明
此函数会返回从公元 1970 年 1 月 1 日的 UTC 时间从 0 时 0 分 0 秒算起到现在所经过的秒数。如果 t 并非空指针的话,此函数也会将返回值存到 t 指针所指的内存。

返回值
成功则返回秒数,失败则返回 ((time_t)-1) 值,错误原因存于 errno 中。

示例

  1. #include <stdio.h>
  2. #include <time.h>
  3. int main(void)
  4. {
  5. long seconds = time((time_t *)NULL);
  6. printf("%ld\n", seconds);
  7. return 0;
  8. }

gettimeofday

函数定义
int gettimeofday (struct timeval * tv, struct timezone * tz)

函数说明
函数gettimeofday()可获得微妙级(0.000001 秒)的系统时间,调用两次 gettimeofday(),前后做减法,从而达到定时或者计算时间的目的。

参数说明

  1. struct timeval{
  2.   long tv_sec; //秒
  3.   long tv_usec; //微秒
  4. };
  5. struct timezone
  6. {
  7.   int tz_minuteswest; //和Greenwich 时间差了多少分钟
  8.   int tz_dsttime; //日光节约时间的状态
  9. };

返回值
该函数成功时返回0,失败时返回-1

char asctime(const struct tm tm);
char asctime_r(const struct tm tm, char buf);
char
ctime(const time_t timep);
char
ctime_r(const time_t timep, char buf);
struct tm gmtime(const time_t timep); // 获取的为英国时间
struct tm gmtime_r(const time_t timep, struct tm result);
struct tm
localtime(const time_t timep); // 获取的为本地时间, 注意与英国时间的区别。
struct tm
localtime_r(const time_t timep, struct tm result);
time_t mktime(struct tm tm);
double difftime(time_t time1, time_t time0);
int gettimeofday(struct timeval
tv, struct timezone tz);
int settimeofday(const struct timeval
tv , const struct timezone *tz);

示例

  1. #include <stdio.h>
  2. #include <sys/time.h>
  3. #include <unistd.h>
  4. int main()
  5. {
  6. struct timeval tv;
  7. gettimeofday(&tv, NULL);
  8. printf("second: %ld\n", tv.tv_sec); // 秒
  9. printf("millisecond: %ld\n", tv.tv_sec * 1000 + tv.tv_usec / 1000); // 毫秒
  10. printf("microsecond: %ld\n", tv.tv_sec * 1000000 + tv.tv_usec); // 徽秒
  11. sleep(3); // 让程序休眠3秒
  12. printf("---------------------sleep 3 second-------------------\n");
  13. gettimeofday(&tv, NULL);
  14. printf("second: %ld\n", tv.tv_sec); // 秒
  15. printf("millisecond: %ld\n",
  16. tv.tv_sec * 1000 + tv.tv_usec / 1000); // 毫秒
  17. printf("microsecond: %ld\n", tv.tv_sec * 1000000 + tv.tv_usec); // 徽秒
  18. return 0;
  19. }

运行结果:

  1. second: 1659441664
  2. millisecond: 1659441664553
  3. microsecond: 1659441664553395
  4. ---------------------sleep 3 second-------------------
  5. second: 1659441667
  6. millisecond: 1659441667553
  7. microsecond: 1659441667553520

第二章 计算时间差

clock_gettime

函数定义
在 POSIX1003.1 中增添了这个函数, 它的原型如下:
int clock_gettime(clockid_t clk_id, struct timespec *tp);

函数说明
实时函数 clock_gettime 单位是十亿分之一秒,也就是纳秒 (ns),使用的是标准 POSIX 实时时钟, 计算出来的结果可能有误差。

参数说明
struct timespec ,timespec 计算时间次数的单位是十亿分之一秒.

  1. struct timespec{
  2. time_t tv_sec;/* Seconds. */
  3. long tv_nsec;/* Nanoseconds. */
  4. }

clockid_t 是确定哪个时钟类型:
CLOCK_REALTIME: 标准 POSIX 实时时钟
CLOCK_MONOTONIC: POSIX 时钟, 以恒定速率运行; 不会复位和调整, 它的取值和 CLOCK_REALTIME 是一样的.
CLOCK_PROCESS_CPUTIME_ID 和 CLOCK_THREAD_CPUTIME_ID 是 CPU 中的硬件计时器中实现的.

示例

  1. #include <stdio.h>
  2. #include <stdlib.h>
  3. #include <time.h>
  4. #include <unistd.h>
  5. #define BILLION 1000 * 1000 * 1000L
  6. int main(void)
  7. {
  8. long int loop = 1000;
  9. struct timespec ts_start;
  10. struct timespec ts_end;
  11. long time_span;
  12. clock_gettime(CLOCK_MONOTONIC, &ts_start);
  13. sleep(1);
  14. clock_gettime(CLOCK_MONOTONIC, &ts_end);
  15. time_span = BILLION * (ts_end.tv_sec - ts_start.tv_sec) +
  16. ts_end.tv_nsec - ts_start.tv_nsec;
  17. fprintf(stdout, "it took %ld nanoseconds\n", time_span);
  18. return 0;
  19. }

编译:
gcc test3.c -lrt -o test3

计算时间:
time ./test3
it took 1000060679 nanoseconds

gettimeofday

int gettimeofday(struct timeval tv,struct timezonetz), 会把目前的时间 tv 所指的结构返回,当地时区的信息则放到 tz 所指的结构中。这两个结构都放在 / usr/include/sys/time.h 中。 单位 微秒 (μs)

示例

  1. #include <assert.h>
  2. #include <stdio.h>
  3. #include <stdlib.h>
  4. #include <unistd.h>
  5. #include <sys/time.h>
  6. #define MILLION 1000 * 1000L
  7. int main(void)
  8. {
  9. long time_span = 0;
  10. struct timeval start;
  11. struct timeval end;
  12. //struct timezone tz; // 后面有说明
  13. // 注意:gettimeofday需要头文件#include <sys/time.h>
  14. gettimeofday(&start, NULL); //gettimeofday(&start,&tz); 结果一样
  15. printf("start.tv_sec:%ld\n", start.tv_sec);
  16. printf("start.tv_usec:%ld\n", start.tv_usec);
  17. sleep(3);
  18. gettimeofday(&end, NULL);
  19. printf("end.tv_sec:%ld\n", end.tv_sec);
  20. printf("end.tv_usec:%ld\n", end.tv_usec);
  21. time_span = (end.tv_sec - start.tv_sec) * MILLION +
  22. (end.tv_usec - start.tv_usec); // 微秒
  23. printf("time span is %ld microseconds\n", time_span);
  24. return 0;
  25. }

输出
start.tv_sec:1659491954
start.tv_usec:582517
end.tv_sec:1659491957
end.tv_usec:582629
time span is 3000112 microseconds

下面的采用指针的方式也可以,但是要注意指针类型若不分配内存的话,编译正确,但是运行结果会不对

times

include
clock_t times(struct tms *buf); 单位 10 毫秒 (ms) times 实际上面就是调用 clock() 实现的。

times() 函数返回从过去一个任意的时间点所经过的时钟数。返回值可能会超出 clock_t (一般为 long 型) 的范围 (溢出)。如果发生错误,则返回 (clock_t ) -1 类型,然后设置相应的 errno 值。

系统每秒的时钟可以通过 sysconf(_SC_CLK_TCK); 函数获得。

tms 结构体如下:
strace tms{
clock_t tms_utime;
clock_t tms_stime;
clock_t tms_cutime;
clock_t tms_cstime;
}

注释:
tms_utime 记录的是进程运行结果用户代码的时间.
tms_stime 记录的是进程运行结果内核代码的时间.
tms_cutime 记录的是子进程运行结果用户代码的时间.
tms_cstime 记录的是子进程运行结果内核代码的时间.

  1. 测试:

vi test2.c

  1. #include <sys/times.h>
  2. #include <stdio.h>
  3. #include <stdlib.h>
  4. #include <sys/types.h>
  5. #include <unistd.h>
  6. static void do_cmd(char *);
  7. static void pr_times(clock_t, struct tms *, struct tms *);
  8. int main(int argc, char *argv[])
  9. {
  10. int i;
  11. for (i = 1; argv[i] != NULL; i++) {
  12. do_cmd(argv[i]);
  13. }
  14. exit(1);
  15. }
  16. static void do_cmd(char *cmd)
  17. {
  18. struct tms tmsstart, tmsend;
  19. clock_t start, end;
  20. int status;
  21. if ((start = times(&tmsstart)) == -1)
  22. puts("times error");
  23. if ((status = system(cmd)) < 0)
  24. puts("system error");
  25. if ((end = times(&tmsend)) == -1)
  26. puts("times error");
  27. pr_times(end - start, &tmsstart, &tmsend);
  28. exit(0);
  29. }
  30. static void pr_times(clock_t real, struct tms *tmsstart, struct tms *tmsend)
  31. {
  32. static long clktck = 0;
  33. if (0 == clktck)
  34. if ((clktck = sysconf(_SC_CLK_TCK)) < 0)
  35. puts("sysconf err");
  36. printf("real:%7.2f\n", real / (double)clktck);
  37. printf("user-cpu:%7.2f\n",
  38. (tmsend->tms_utime - tmsstart->tms_utime) / (double)clktck);
  39. printf("system-cpu:%7.2f\n",
  40. (tmsend->tms_stime - tmsstart->tms_stime) / (double)clktck);
  41. printf("child-user-cpu:%7.2f\n",
  42. (tmsend->tms_cutime - tmsstart->tms_cutime) / (double)clktck);
  43. printf("child-system-cpu:%7.2f\n",
  44. (tmsend->tms_cstime - tmsstart->tms_cstime) / (double)clktck);
  45. }

编译:
gcc test2.c -o test2

测试这个程序:
time ./test2 “dd if=/dev/zero f=/dev/null bs=1M count=10000”
10000+0 records in
10000+0 records out
10485760000 bytes (10 GB) copied, 4.93028 s, 2.1 GB/s
real: 4.94
user-cpu: 0.00
system-cpu: 0.00
child-user-cpu: 0.01
child-system-cpu: 4.82

clock

#include
clock_t clock(void); 单位 10 毫秒 (ms)
int clock_gettime(clockid_t clk_id, struct timespec *tp);
系统每秒的时钟可以通过 sysconf(_SC_CLK_TCK); 函数获得。

以下是各种精确度的类型转换:
1 秒 = 1000 毫秒 (ms), 1 毫秒 = 1/1000 秒 (s);
1 秒 = 1000000 微秒 (μs), 1 微秒 = 1/1000000 秒 (s);
1 秒 = 1000000000 纳秒 (ns),1 纳秒 = 1/1000000000 秒 (s);

clock() 函数的精确度是 10 毫秒 (ms)
times() 函数的精确度是 10 毫秒 (ms)
gettimofday() 函数的精确度是微秒 (μs)
clock_gettime() 函数的计量单位为十亿分之一,也就是纳秒 (ns)

3) times() 和 clock()

默认的 Linux 时钟周期是 100HZ, 而现在最新的内核时钟周期默认为 250HZ.
如何得到内核的时钟周期呢?
grep ^CONFIG_HZ /boot/config-2.6.26-1-xen-amd64

CONFIG_HZ_250=y
CONFIG_HZ=250

结果就是 250HZ.

而用 sysconf(_SC_CLK_TCK); 得到的却是 100HZ
例如:

  1. #include <stdio.h>
  2. #include <stdlib.h>
  3. #include <unistd.h>
  4. // #include <time.h>
  5. // #include <sys/time.h>
  6. int main(int argc, char *argv[])
  7. {
  8. long tps = sysconf(_SC_CLK_TCK);
  9. printf("%ld\n", tps);
  10. return EXIT_SUCCESS;
  11. }

为什么得到的是不同的值呢?
因为 sysconf(_SC_CLK_TCK) 和 CONFIG_HZ 所代表的意义是不同的.
sysconf(_SC_CLK_TCK) 是 GNU 标准库的 clock_t 频率.
它的定义位置在:/usr/include/asm/param.h

例如:
#ifndef HZ
#define HZ 100
#endif

最后总结一下内核时间:
内核的标准时间是 jiffy, 一个 jiffy 就是一个内部时钟周期, 而内部时钟周期是由 250HZ 的频率所产生中的, 也就是一个时钟滴答, 间隔时间是 4 毫秒 (ms).

也就是说:
1 个 jiffy=1 个内部时钟周期 = 250HZ=1 个时钟滴答 = 4 毫秒

sysconf(_SC_CLK_TCK) 使用默认的 Linux 时钟周期是 100HZ, 1 个 jiffy=1 个内部时钟周期 = 100HZ=1 个时钟滴答 = 10 毫秒,所以 clock() 和 times()的最新单位是 10ms

每经过一个时钟滴答就会调用一次时钟中断处理程序,处理程序用 jiffy 来累计时钟滴答数, 每发生一次时钟中断就增 1.
而每个中断之后, 系统通过调度程序跟据时间片选择是否要进程继续运行, 或让进程进入就绪状态.

https://blog.csdn.net/sunny04/article/details/41848295