Unix/Linux时间函数和日历函数

本文详细介绍了Unix/Linux系统中时间处理的核心概念,包括时间结构体、精度、日历结构以及获取系统时间的方法,如time、timeval、timespec等。同时,阐述了如何在代码中使用这些时间结构进行高精度计算,以及如何将time_t转换为本地或国际标准时间,最终通过日期格式化函数实现日期时间的显示。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

Unix/Linux都是采用UTC(Universal Coordinated Time),1970.1.1到现在的秒数,采用time_t(long int)存储

1. 时间结构体

都是定义在time.h头文件中

时间:time_t, timeval, time_spec, 精度越来越高

日历:tm


1.1 time_t

typedef long time_t

time_t表示为从UTC(coordinated universal time)时间197011000000(也称为Linux系统的Epoch时间)到当前时刻的秒数。


1.2 timeval
struct timeval
{
	time_t 	tv_sec; //秒 s
	long 	tv_usec;	//微秒us
};

1.3 timespec

struct timespec
{
	long int tv_sec;	//秒 s
	long int tv_nsec;	//纳秒 ns
};

1.4 tm

struct tm{  
    int tm_sec; /*秒 - 取值区间为[0, 59]*/  
    int tm_min; /*分 - 取值区间为[0, 59]*/  
    int tm_hour; /*时 - 取值区间为[0, 23]*/  
    int tm_mday; /*日 - 取值区间为[1, 31]*/  
    int tm_mon; /*月份 - 取值区间为[0, 11]*/  
    int tm_year; /*年份 - 其值为1900年至今年数*/  
    int tm_wday; /*星期 - 取值区间[0, 6],0代表星期天,1代表星期1,以此类推*/  
    int tm_yday; /*从每年的1月1日开始的天数-取值区间为[0, 365],0代表1月1日*/  
    int tm_isdst; /*夏令时标识符,使用夏令时,tm_isdst为正,不使用夏令时,tm_isdst为0,不了解情况时,tm_isdst为负*/  
	long int tm_gmtoff;	
	const char* tm_zone;
};  

前9个成员是必须成员,最后2个成员根据系统会有不同。

tm_gmtoff:指定了日期变更线东面时区中UTC东部时区正秒数或UTC西部时区的负秒数

tm_zone:当前时区的名字(与环境变量TZ有关)


2. 获取系统时间

2.1 time_t time( time_t *t );

t和返回值都返回UTC时间,t可以为空

	time_t cur_time;

	time(&cur_time);
	cur_time = time(NULL);

	if ( cur_time == -1 )
	{
		perror("time");
		exit(1);
	}


	printf("Current time : %d\n", cur_time);

double difftime( time_t time1 , time_t time2 );

求time1和time2 的差值

2.2 gettimeofday 获取更高的精度

头文件:sys/time.h 和unistd.h

int gettimeofday(struct timeval*tv, struct timezone *tz);

struct timezone{
int tz_minuteswest;/*格林威治时间往西方的时差*/
int tz_dsttime;/*DST 时间的修正方式*/
};
gettimeofday( )把目前的时间信息存入tv指向的结构体,当地时区信息则放到tz指向的结构体。

3. timeval 运算

在sys/time.h中还定义了5个宏对timeval 进行算数运算

/*判断tvp是否填充*/
#define       timerisset(tvp)  ((tvp)->tv_sec || (tvp)->tv_usec) 

/*tvp设置为0*/
#define       timerclear(tvp) ((tvp)->tv_sec = (tvp)->tv_usec = 0)  

/*tvp比较,cmp不可为>=, <=*/
#define       timercmp(tvp, uvp, cmp) 
              ((tvp)->tv_sec cmp (uvp)->tv_sec ||\ 
               (tvp)->tv_sec == (uvp)->tv_sec &&\ 
               (tvp)->tv_usec cmp (uvp)->tv_usec) 
			   
/*result = a+b*/
#define		timeradd(a, b, result)

/*result = a-b*/
#define		timersub(a, b, result)


4. time_t -->tm

4.1 struct tm* gmtime( const time_t* p_time );   

头文件:time.h

返回:国际标准时间(GMT)

int main(void)
{
    char *wday[] = {"Sun", "Mon", "Tue", "Wed", "Thu", "Fri", "Sat"};
    time_t timep;
    struct tm *p_tm;
    timep = time(NULL);
    p_tm = gmtime(&timep); /*获取GMT时间*/
    printf("%d-%d-%d ", (p_tm->tm_year+1900), (p_tm->mon+1), p_tm->tm_mday);
    printf("%s %d:%d:%d\n", wday[p_tm->tm_wday], p_tm->tm_hour, p_tm->tm_min, p_tm->tm_sec);
}


4.2 struct tm* localtime( const time_t* p_time );

头文件:time.h

返回:本地时间(受时区影响,与环境变量TZ有关)

int main(void)
{
    char *wday[] = {"Sun", "Mon", "Tue", "Wed", "Thu", "Fri", "Sat"};
    time_t timep;
    struct tm *p_tm;
    timep = time(NULL);
    p_tm = localtime(&timep); /*获取本地时区时间*/
    printf("%d-%d-%d ", (p_tm->tm_year+1900), (p_tm->mon+1), p_tm->tm_mday);
    printf("%s %d:%d:%d\n", wday[p_tm->tm_wday], p_tm->tm_hour, p_tm->tm_min, p_tm->tm_sec);
    return 0;
}

5. tm --> time_t

头文件:time.h

time_t mktime( struct tm* p_tm );


6. time --> char*

6.1 asctime

头文件:#include <time.h>
函数定义:char *asctime(const struct tm *p_tm);
功能描述:asctime( )将参数p_tm指向的tm结构体数据转换成实际使用的时间日期表示方法,并以字符串形式返回(ctime函数相同)。字符串格式为:"Wed Jun 20 21:00:00 2012\n"

6.2 ctime

同asctime

ctime一率使用当地时间,asctime则用tm结构内的timezone资讯来表示。

7. 格式化 strftime strptime

7.1 size_t strftime(char *str,size_t max,char *fmt,struct tm *tp);

strftime有点像sprintf,其格式由fmt来指定。

%a : 本第几天名称,缩写。
%A : 本第几天名称,全称。
%b : 月份名称,缩写。
%B : 月份名称,全称。
%c : 与ctime/asctime格式相同。
%d : 本月第几日名称,由零算起。
%H : 当天第几个小时,24小时制,由零算起。
%I : 当天第几个小时,12小时制,由零算起。
%j : 当年第几天,由零算起。
%m : 当年第几月,由零算起。
%M : 该小时的第几分,由零算起。
%p : AM或PM。
%S : 该分钟的第几秒,由零算起。
%U : 当年第几,由第一个日开始计算。
%W : 当年第几,由第一个一开始计算。
%w : 当第几日,由零算起。
%x : 当地日期。
%X : 当地时间。
%y : 两位数的年份。
%Y : 四位数的年份。
%Z : 时区名称的缩写。
%% : %符号。


7.2 char * strptime(char *s,char *fmt,struct tm *tp);

如同scanf一样,解译字串成为tm格式。

%h : 与%b及%B同。
%c : 读取%x及%X格式。
%C : 读取%C格式。
%e : 与%d同。
%D : 读取%m/%d/%y格式。
%k : 与%H同。
%l : 与%I同。
%r : 读取"%I:%M:%S %p"格式。
%R : 读取"%H:%M"格式。
%T : 读取"%H:%M:%S"格式。
%y : 读取两位数年份。
%Y : 读取四位数年份。 


### Linux 时间函数概述 在Linux系统中,时间处理是一个非常常见的需求。C标准库提供了多种与时间相关的函数来满足不同的应用场景。以下是几个常用的与时间有关的函数及其使用方法。 #### `time()` 函数 `time()` 是用来获取当前日历时间的一个简单函数。它返回的是自1970年1月1日以来经过的秒数(即Unix时间戳)。 ```c #include <stdio.h> #include <time.h> int main() { time_t now; time(&now); printf("Current Unix timestamp: %ld\n", (long)now); return 0; } ``` 此代码展示了如何调用 `time()` 并打印出当前的时间戳[^1]。 #### `localtime()` `gmtime()` 函数 这两个函数分别用于将时间戳转换为本地时间世界协调时间(UTC)。它们都接受一个指向 `time_t` 的指针作为参数,并返回一个指向 `struct tm` 结构体的指针。 ```c #include <stdio.h> #include <time.h> int main() { time_t raw_time; struct tm *info; time(&raw_time); info = localtime(&raw_time); // 获取本地时间 printf("Local Time: %d-%02d-%02d %02d:%02d:%02d\n", info->tm_year + 1900, info->tm_mon + 1, info->tm_mday, info->tm_hour, info->tm_min, info->tm_sec); info = gmtime(&raw_time); // 获取 UTC 时间 printf("UTC Time: %d-%02d-%02d %02d:%02d:%02d\n", info->tm_year + 1900, info->tm_mon + 1, info->tm_mday, info->tm_hour, info->tm_min, info->tm_sec); return 0; } ``` 这段程序演示了如何通过 `localtime()` `gmtime()` 来获得并格式化输出日期时间。 #### `clock()` 函数 `clock()` 可以测量CPU执行某个操作所花费的实际时间。其单位通常是处理器时钟周期的数量除以常量 `CLOCKS_PER_SEC` 转换成秒。 ```c #include <stdio.h> #include <stdlib.h> #include <time.h> int main() { long i = 10000000L; clock_t start, finish; double duration; start = clock(); while (i--) ; finish = clock(); duration = ((double)(finish - start)) / CLOCKS_PER_SEC; printf("Loop took %.6f seconds.\n", duration); return 0; } ``` 上述例子说明了怎样利用 `clock()` 计算一段空循环耗时的情况[^3]。 #### `strftime()` 函数 当需要把结构化的日期信息转化为字符串形式展示给用户看的时候可以考虑使用 `strftime()` 。它可以按照指定格式生成可读性强的人类友好的表示方式。 ```c #include <stdio.h> #include <time.h> int main(){ char buffer[80]; time_t t = time(NULL); struct tm tm_info = *localtime(&t); strftime(buffer,sizeof(buffer),"%A %B %e %Y %H:%M:%S",&tm_info); puts(buffer); return 0; } ``` 这里给出了关于如何借助于 `strftime()` 定制输出样式的示范。 --- ###
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值