C语言标准库 time.h
C语言标准库 time.h
Diebugtime.h
time_t
time_t 是一个表示时间的类型别名,可以视为国际标准时 UTC。它可能是浮点数,也可能是整数,Unix 系统一般是整数。
许多系统上,time_t 表示自时间纪元(time epoch)以来的秒数。Unix 的时间纪元是国际标准时 UTC 的1970年1月1日的零分零秒。time_t 如果为负数,则表示时间纪元之前的时间。
time_t
一般是32位或64位整数类型的别名,具体类型取决于当前系统。如果是32位带符号整数,time_t
可以表示的时间到 2038年1月19日03:14:07 UTC
为止;如果是32位无符号整数,则表示到2106年。如果是64位带符号整数,可以表示-2930
亿年到+2930
亿年的时间范围。
struct tm
struct tm 是一个数据结构,用来保存时间的各个组成部分,比如小时、分钟、秒、日、月、年等。下面是它的结构。
1 | struct tm { |
time()
time()
函数返回从时间纪元到现在经过的秒数。
1 | time_t time(time_t* returned_value); |
time()
接受一个 time_t
指针作为参数,返回值会写入指针地址。参数可以是空指针 NULL。
time()
的返回值是 time_t 类型的当前时间。
如果计算机无法提供当前的秒数,或者返回值太大,无法用time_t
类型表示,time()
函数就返回-1
。
1 | time_t now; |
上面示例展示了将当前时间存入变量now
的两种写法。
如果要知道某个操作耗费的精确时间,需要调用两次time()
,再将两次的返回值相减。
1 | time_t begin = time(NULL); |
注意,上面的方法只能精确到秒。
ctime()
ctime()
用来将 time_t
类型的值直接输出为人类可读的格式。
1 | char* ctime( time_t const * time_value ); |
ctime()
的参数是一个 time_t
指针,返回一个字符串指针。该字符串的格式类似“Sun Jul 4 04:02:48
1976\n\0”,尾部包含换行符和字符串终止标志。
下面是一个例子。
1 | time_t now; |
注意,ctime()
会在字符串尾部自动添加换行符。
localtime(),gmtime()
localtime()
函数用来将 time_t
类型的时间,转换为当前时区的 struct tm 结构。
gmtime()
函数用来将 time_t 类型的时间,转换为 UTC 时间的
struct tm 结构。
它们的区别就是返回值,前者是本地时间,后者是 UTC 时间。
1 | struct tm* localtime(const time_t* timer); |
下面是一个例子。
1 | time_t now = time(NULL); |
asctime()
asctime()
函数用来将 struct tm
结构,直接输出为人类可读的格式。该函数会自动在输出的尾部添加换行符。
用法示例参考上一小节。
mktime()
mktime()
函数用于把一个 struct tm 结构转换为 time_t
值。
1 | time_t mktime(struct tm* tm_ptr); |
mktime()
的参数是一个 struct tm 指针。
mktime()
会自动设置 struct tm
结构里面的tm_wday
属性和tm_yday
属性,开发者自己不必填写这两个属性。所以,这个函数常用来获得指定时间是星期几(tm_wday
)。
struct tm
结构的tm_isdst
属性也可以设为-1
,让mktime()
决定是否应该采用夏令时。
下面是一个例子。
1 | struct tm some_time = { |
difftime()
difftime()
用来计算两个时间之间的差异。Unix
系统上,直接相减两个 time_t
值,就可以得到相差的秒数,但是为了程序的可移植性,最好还是使用这个函数。
1 | double difftime( time_t time1, time_t time2 ); |
difftime()
函数接受两个 time_t 类型的时间作为参数,计算
time1 - time2 的差,并把结果转换为秒。
注意它的返回值是 double 类型。
1 | #include <stdio.h> |
上面示例中,折算年份时,为了尽量准确,使用了一年的准确长度 365.2425 天,这样可以抵消闰年的影响。
strftime()
strftime()
函数用来将 struct tm
结构转换为一个指定格式的字符串,并复制到指定地址。
1 | size_t strftime( |
strftime()
接受四个参数。
- 第一个参数:目标字符串的指针。
- 第二个参数:目标字符串可以接受的最大长度。
- 第三个参数:格式字符串。
- 第四个参数:struct tm 结构。
如果执行成功(转换并复制),strftime()
函数返回复制的字符串长度;如果执行失败,返回-1
。
下面是一个例子。
1 | #include <stdio.h> |
下面是常用的格式占位符。
- %%:输出 % 字符。
- %a:星期几的简写形式,以当地时间计算。
- %A:星期几的完整形式,以当地时间计算。
- %b:月份的简写形式,以当地时间计算。
- %B:月份的完整形式,以当地时间计算。
- %c:日期和时间,使用“%x %X”。
- %d:月份的天数(01-31)。
- %H:小时,采用24小时制(00-23)。
- %I:小时,采用12小时制(00-12)。
- %J:一年的第几天(001-366)。
- %m:月数(01-12)。
- %M:分钟(00~59)。
- %P:AM 或 PM。
- %R:相当于“%H:%M“。
- %S:秒(00-61)。
- %U:一年的第几星期(00-53),以星期日为第1天。
- %w:一星期的第几天,星期日为第0天。
- %W:一年的第几星期(00-53),以星期一为第1天。
- %x:完整的年月日的日期,以当地时间计算。
- %X:完整的时分秒的时间,以当地时间计算。
- %y:两位数年份(00-99)。
- %Y:四位数年份(例如 1984)。
- %Z:时区的简写。
timespec_get()
timespec_get()
用来将当前时间转成距离时间纪元的纳秒数(十亿分之一秒)。
1 | int timespec_get ( struct timespec* ts, int base ) ; |
timespec_get()
接受两个参数。
第一个参数是 struct timespec 结构指针,用来保存转换后的时间信息。struct timespec 的结构如下。
1 | struct timespec { |
第二个参数是一个整数,表示时间计算的起点。标准只给出了宏 TIME_UTC 这一个可能的值,表示返回距离时间纪元的秒数。
下面是一个例子。
1 | struct timespec ts; |
clock()
clock()
函数返回从程序开始执行到当前的 CPU
时钟周期。一个时钟周期等于 CPU 频率的倒数,比如 CPU 的频率如果是 1G
Hz,表示1秒内时钟信号可以变化 109 次,那么每个时钟周期就是
10-9 秒。
1 | clock_t clock(void); |
clock()
函数返回一个数字,表示从程序开始到现在的 CPU
时钟周期的次数。这个值的类型是 clock_t,一般是 long int 类型。
为了把这个值转换为秒,应该把它除以常量CLOCKS_PER_SEC
(每秒的时钟周期),这个常量也由time.h
定义。
1 | printf("CPU time: %f\n", clock() / (double)CLOCKS_PER_SEC); |
上面示例可以输出程序从开始到运行到这一行所花费的秒数。
如果计算机无法提供 CPU
时间,或者返回值太大,无法用clock_t
类型表示,clock()
函数就返回-1
。
为了知道某个操作所耗费的精确时间,需要调用两次clock()
,然后将两次的返回值相减。
1 | clock_t start = clock(); |