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)时间1970年1月1日00时00分00秒(也称为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 : 读取四位数年份。