时间转换函数

30 篇文章 1 订阅
13 篇文章 0 订阅
一、定义自定义的时间格式MYtime
  struct tm使用不方便,自定义时间格式MYtime方便显示。
 struct MYtime
{
    int m_iSec;     /* = int tm_sec;    tm seconds after the minute - [0,59] */
    int m_iMin;     /* = int tm_min;    tm minutes after the hour - [0,59] */
    int m_iHour; /* = int tm_hour;   tm hours since midnight - [0,23] */
    int m_iDay;     /* = int tm_mday;   tm  day of the month - [1,31] */
    int m_iMon;     /* m_iMon = tm_mon+1 int tm_mon;    tm months since January - [0,11] */
    int m_iYear; /* m_iYear = tm_year + 1900 int tm_year;   tm years since 1900 */
         /* int tm_wday;   tm days since Sunday - [0,6] */
         /* int tm_yday;   tm days since January 1 - [0,365] */
    int m_iIsdst; /* m_iIsdst= int tm_isdst;  tm daylight savings time flag */
    MYtime()
    {
        m_iSec = 0;     
        m_iMin = 0;          
        m_iHour = 0;     
        m_iDay = 0;      
        m_iMon = 0;      
        m_iYear = 0;     
        m_iIsdst = 0;
    };
    MYtime(int iYear, int iMon, int iDay,int iHour,int iMin,int iSec = 0)
    {
        m_iSec = iSec;     
        m_iMin = iMin;          
        m_iHour = iHour;     
        m_iDay = iDay;      
        m_iMon = iMon;      
        m_iYear = iYear;     
        m_iIsdst = 0;
    };


};


二、MYtime转化为tm格式
};
//MYtime转化为tm格式
void MYtimeToTM(const MYtime& oMytime, struct tm& oTM)
{
    oTM.tm_sec  = oMytime.m_iSec;   
    oTM.tm_min = oMytime.m_iMin;    
    oTM.tm_hour = oMytime.m_iHour;  
    oTM.tm_mday = oMytime.m_iDay;   
    oTM.tm_mon = oMytime.m_iMon - 1 ;   
    oTM.tm_year = oMytime.m_iYear - 1900 ;
    oTM.tm_isdst = oMytime.m_iIsdst;
}
//tm格式转换为MYtime
void TMToMYtime(const struct tm& oTM,MYtime& oMytime)
{


    oMytime.m_iSec = oTM.tm_sec;   
    oMytime.m_iMin = oTM.tm_min;    
    oMytime.m_iHour = oTM.tm_hour;  
    oMytime.m_iDay = oTM.tm_mday;   
    oMytime.m_iMon = oTM.tm_mon + 1;   
    oMytime.m_iYear = oTM.tm_year + 1900;
    oMytime.m_iIsdst= oTM.tm_isdst;


}


三、定义固定格式的时间string,方便存储
 为了和上面的方便,定义两个转化函数
bool MYtimeToStr(const MYtime &oMytime,std::string& strTime)
{
    char szBuf[20] = { 0};//时间最多不超过20位
    int iRet = sprintf(szBuf, "%4d-%2d-%2d %2d:%2d",     
        oMytime.m_iYear,
        oMytime.m_iMon,
        oMytime.m_iDay,
        oMytime.m_iHour,
        oMytime.m_iMin);
    if(iRet < 5)//表示时间格式错,不能正确地读取到秒??m_iSec
    {
        return false;
    }
    strTime = szBuf;
    return true;
}
四、好多时间格式,还分GMT、Local,好乱,理一理


 /时间转化的顺序:  MYtimeToStr           MYtime(可理解的) <--> String(可理解的)    StrToMYtime
                    MYtimeToTM            MYtime<-->struct tm                           TMToMYtime
                    mktime                struct tm <--->time_t(long)                 gmtime,localtime 
                    struct tm -->mktime-->time_t(local)
                    struct tm -->mktime--> -_timezone --> time_t(gmt?)


五、根据使用场景定义的两组转化函数
//场景一:
//数据库和传输时使用time_t(gmt) 时间,界面显示使用MYtime(local)
//界面显示<-->数据库和传输:    MYtime(local) <---> struct tm(local)    <--->  time_t(long) 
//                                MYtimeToTM-->mktime
//                                TMToMYtime<--localtime


bool LOCMytimeToGMTCalender(const MYtime& oLOCMytime,time_t &oGMTCalender)
{
    struct tm oLOCTM;
    MYtimeToTM(oLOCMytime,oLOCTM);
    oGMTCalender = mktime(&oLOCTM);
    if(-1 == oGMTCalender)
    {
        return false;
    }
    return true;
}








bool GMTCalenderToLOCMytime(const time_t oGMTCalender, MYtime& oLOCMytime)
{
    struct tm *pLOCTM = NULL;
    pLOCTM = localtime(&oGMTCalender);
    if(NULL == pLOCTM)
    {
        return false;
    }
    TMToMYtime(*pLOCTM,oLOCMytime);
    return true;
}






//场景二:
//数据库和传输是使用String(gmt)时间,界面显示使用MYtime(local)
//界面显示<-->数据库和传输:    MYtime(local) <---> struct tm(local)<---> time_t(long) <---> struct tm(gmt)<--->  MYtime(gmt)<-->String(gmt)
//                                            MYtimeToTM        -->         mktime            -->  gmtime        -->        TMToMYtime    -->      MYtimeToStr
//                                            TMToMYtime        <--         localtime        <--  -_timezone mktime        <--        MYtimeToTM    <--      StrToMYtime


bool GMTStrToLOCMytime(const std::string& strGMTStr, MYtime& oLOCMytime)
{
    MYtime oGMTMytime;
    if(!StrToMYtime(strGMTStr,oGMTMytime))
    {
        return false;
    }
    struct tm oGMTTM;
    MYtimeToTM(oGMTMytime, oGMTTM);


    time_t lLocTime = mktime(&oGMTTM);
    if(-1 == lLocTime)
    {
        return false;
    }
    lLocTime -= _timezone;//很关键


    struct tm *pLOCTM = NULL;
    pLOCTM = localtime(&lLocTime);
    if(NULL == pLOCTM)
    {
        return false;
    }
    TMToMYtime(*pLOCTM,oLOCMytime);
    return true;
}




bool LOCMytimeToGMTStr(const MYtime& oLOCMytime,std::string& strGMTStr)
{
    struct tm oLOCTM;
    MYtimeToTM(oLOCMytime,oLOCTM);


    time_t lTime = mktime(&oLOCTM);
    if(-1 == lTime)
    {
        return false;
    }
    struct tm *pGMTTM = NULL;
    pGMTTM = gmtime(&lTime);
    if(NULL == pGMTTM)
    {
        return false;
    }
    MYtime oGMTMYtime;
    TMToMYtime(*pGMTTM,oGMTMYtime);


    if(!MYtimeToStr(oGMTMYtime,strGMTStr))
    {
        return false;
    }
    return true;
}


六、附带的单元测试
int _tmain(int argc, _TCHAR* argv[])
{
    //std::string strTime="2008-8-8 12:33";
    //std::string strTime="2008-08-08 12:33";
    //std::string strTime="2008-d8-08 12:33";
    std::string strTime="2008-8-08 12:33";
    MYtime mytm;
    StrToMYtime(strTime,mytm);


    //MYtime oMytime2(2008,8,8,12,33);
    MYtime oMytime2(308,8,8,12,33);
    std::string strTime2;
    MYtimeToStr(oMytime2,strTime2);


    //场景二的两个时间转化函数
    std::string strGMTStr = "2008-8-8 12:33";
    MYtime oLOCMytime;
    GMTStrToLOCMytime(strGMTStr, oLOCMytime);


    std::string strGMTStr2;
    MYtime oLOCMytime2(2008,8,8,12,33);
    LOCMytimeToGMTStr(oLOCMytime2 ,strGMTStr2);




    //场景一
    MYtime oLOCMytime3(2008,8,8,12,33);
    time_t oGMTCalender=0;
    LOCMytimeToGMTCalender(oLOCMytime3, oGMTCalender);


    time_t oGMTCalender2=0;
    oGMTCalender2 = time(NULL);
    MYtime oLOCMytime4;
    GMTCalenderToLOCMytime(oGMTCalender2,oLOCMytime4 );


    return 0;
}


本文从介绍基础概念入手,探讨了在C/C++中对日期和时间操作所用到的数据结构和函数,并对计时、时间的获取、时间的计算和显示格式等方面进行了阐述。本文还通过大量的实例向你展示了time.h头文件中声明的各种函数和数据结构的详细使用方法。


关键字:UTC(世界标准时间),Calendar Time(日历时间),epoch(时间点),clock tick(时钟计时单元)




1.概念
在C/C++中,对字符串的操作有很多值得注意的问题,同样,C/C++对时间的操作也有许多值得大家注意的地方。最近,在技术群中有很多网友也多次问到过C++语言中对时间的操作、获取和显示等等的问题。下面,在这篇文章中,笔者将主要介绍在C/C++中时间和日期的使用方法.


通过学习许多C/C++库,你可以有很多操作、使用时间的方法。但在这之前你需要了解一些“时间”和“日期”的概念,主要有以下几个:


Coordinated Universal Time(UTC):协调世界时,又称为世界标准时间,也就是大家所熟知的格林威治标准时间(Greenwich Mean Time,GMT)。比如,中国内地的时间与UTC的时差为+8,也就是UTC+8。美国是UTC-5。


Calendar Time:日历时间,是用“从一个标准时间点到此时的时间经过的秒数”来表示的时间。这个标准时间点对不同的编译器来说会有所不同,但对一个编译系统来说,这个标准时间点是不变的,该编译系统中的时间对应的日历时间都通过该标准时间点来衡量,所以可以说日历时间是“相对时间”,但是无论你在哪一个时区,在同一时刻对同一个标准时间点来说,日历时间都是一样的。


epoch:时间点。时间点在标准C/C++中是一个整数,它用此时的时间和标准时间点相差的秒数(即日历时间)来表示。


clock tick:时钟计时单元(而不把它叫做时钟滴答次数),一个时钟计时单元的时间长短是由CPU控制的。一个clock tick不是CPU的一个时钟周期,而是C/C++的一个基本计时单位。


我们可以使用ANSI标准库中的time.h头文件。这个头文件中定义的时间和日期所使用的方法,无论是在结构定义,还是命名,都具有明显的C语言风格。下面,我将说明在C/C++中怎样使用日期的时间功能。


2. 计时


C/C++中的计时函数是clock(),而与其相关的数据类型是clock_t。在MSDN中,查得对clock函数定义如下:


clock_t clock( void );


这个函数返回从“开启这个程序进程”到“程序中调用clock()函数”时之间的CPU时钟计时单元(clock tick)数,在MSDN中称之为挂钟时间(wall-clock)。其中clock_t是用来保存时间的数据类型,在time.h文件中,我们可以找到对它的定义:


#ifndef _CLOCK_T_DEFINED
typedef long clock_t;
#define _CLOCK_T_DEFINED
#endif


很明显,clock_t是一个长整形数。在time.h文件中,还定义了一个常量CLOCKS_PER_SEC,它用来表示一秒钟会有多少个时钟计时单元,其定义如下:


#define CLOCKS_PER_SEC ((clock_t)1000)


可以看到可以看到每过千分之一秒(1毫秒),调用clock()函数返回的值就加1。下面举个例子,你可以使用公式clock()/CLOCKS_PER_SEC来计算一个进程自身的运行时间:


void elapsed_time()
{
printf("Elapsed time:%u secs.\n",clock()/CLOCKS_PER_SEC);
}


当然,你也可以用clock函数来计算你的机器运行一个循环或者处理其它事件到底花了多少时间:


#i nclude “stdio.h”
#i nclude “stdlib.h”
#i nclude “time.h”


int main( void )
{
    long     i = 10000000L;
    clock_t start, finish;
    double   duration;
    /* 测量一个事件持续的时间*/
    printf( "Time to do %ld empty loops is ", i );
    start = clock();
    while( i-- )       ;
    finish = clock();
    duration = (double)(finish - start) / CLOCKS_PER_SEC;
    printf( "%f seconds\n", duration );
    system("pause");
}


在笔者的机器上,运行结果如下:


Time to do 10000000 empty loops is 0.03000 seconds


上面我们看到时钟计时单元的长度为1毫秒,那么计时的精度也为1毫秒,那么我们可不可以通过改变CLOCKS_PER_SEC的定义,通过把它定义的大一些,从而使计时精度更高呢?通过尝试,你会发现这样是不行的。在标准C/C++中,最小的计时单位是一毫秒。


3.与日期和时间相关的数据结构


在标准C/C++中,我们可通过tm结构来获得日期和时间,tm结构在time.h中的定义如下:


#ifndef _TM_DEFINED
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代表一月) - 取值区间为[0,11] */
         int tm_year;     /* 年份,其值等于实际年份减去1900 */
         int tm_wday;     /* 星期 – 取值区间为[0,6],其中0代表星期天,1代表星期一,以此类推 */
         int tm_yday;     /* 从每年的1月1日开始的天数 – 取值区间为[0,365],其中0代表1月1日,1代表1月2日,以此类推 */
         int tm_isdst;    /* 夏令时标识符,实行夏令时的时候,tm_isdst为正。不实行夏令时的进候,tm_isdst为0;不了解情况时,tm_isdst()为负。*/
         };
#define _TM_DEFINED
#endif


ANSI C标准称使用tm结构的这种时间表示为分解时间(broken-down time)。


而日历时间(Calendar Time)是通过time_t数据类型来表示的,用time_t表示的时间(日历时间)是从一个时间点(例如:1970年1月1日0时0分0秒)到此时的秒数。在time.h中,我们也可以看到time_t是一个长整型数:


#ifndef _TIME_T_DEFINED
typedef long time_t;          /* 时间值 */
#define _TIME_T_DEFINED      /* 避免重复定义 time_t */
#endif


大家可能会产生疑问:既然time_t实际上是长整型,到未来的某一天,从一个时间点(一般是1970年1月1日0时0分0秒)到那时的秒数(即日历时间)超出了长整形所能表示的数的范围怎么办?对time_t数据类型的值来说,它所表示的时间不能晚于2038年1月18日19时14分07秒。为了能够表示更久远的时间,一些编译器厂商引入了64位甚至更长的整形数来保存日历时间。比如微软在Visual C++中采用了__time64_t数据类型来保存日历时间,并通过_time64()函数来获得日历时间(而不是通过使用32位字的time()函数),这样就可以通过该数据类型保存3001年1月1日0时0分0秒(不包括该时间点)之前的时间。


在time.h头文件中,我们还可以看到一些函数,它们都是以time_t为参数类型或返回值类型的函数:


double difftime(time_t time1, time_t time0);
time_t mktime(struct tm * timeptr);
time_t time(time_t * timer);
char * asctime(const struct tm * timeptr);
char * ctime(const time_t *timer);


此外,time.h还提供了两种不同的函数将日历时间(一个用time_t表示的整数)转换为我们平时看到的把年月日时分秒分开显示的时间格式tm:


struct tm * gmtime(const time_t *timer);                                          
struct tm * localtime(const time_t * timer);


通过查阅MSDN,我们可以知道Microsoft C/C++ 7.0中时间点的值(time_t对象的值)是从1899年12月31日0时0分0秒到该时间点所经过的秒数,而其它各种版本的Microsoft C/C++和所有不同版本的Visual C++都是计算的从1970年1月1日0时0分0秒到该时间点所经过的秒数。


4.与日期和时间相关的函数及应用
在本节,我将向大家展示怎样利用time.h中声明的函数对时间进行操作。这些操作包括取当前时间、算时间间隔、以不同的形式显示时间等内容。


4.1 获得日历时间


我们可以通过time()函数来获得日历时间(Calendar Time),其原型为:


time_t time(time_t * timer);


如果你已经声明了参数timer,你可以从参数timer返回现在的日历时间,同时也可以通过返回值返回现在的日历时间,即从一个时间点(例如:1970年1月1日0时0分0秒)到现在此时的秒数。如果参数为空(NULL),函数将只通过返回值返回现在的日历时间,比如下面这个例子用来显示当前的日历时间:


#i nclude "time.h"
#i nclude "stdio.h"
int main(void)
{
struct tm *ptr;
time_t lt;
lt =time(NULL);
printf("The Calendar Time now is %d\n",lt);
return 0;
}


运行的结果与当时的时间有关,我当时运行的结果是:


The Calendar Time now is 1122707619


其中1122707619就是我运行程序时的日历时间。即从1970年1月1日0时0分0秒到此时的秒数。


4.2 获得日期和时间


这里说的日期和时间就是我们平时所说的年、月、日、时、分、秒等信息。从第2节我们已经知道这些信息都保存在一个名为tm的结构体中,那么如何将一个日历时间保存为一个tm结构的对象呢?


其中可以使用的函数是gmtime()和localtime(),这两个函数的原型为:


struct tm * gmtime(const time_t *timer);                                          
struct tm * localtime(const time_t * timer);


其中gmtime()函数是将日历时间转化为世界标准时间(即格林尼治时间),并返回一个tm结构体来保存这个时间,而localtime()函数是将日历时间转化为本地时间。比如现在用gmtime()函数获得的世界标准时间是2005年7月30日7点18分20秒,那么我用localtime()函数在中国地区获得的本地时间会比时间标准时间晚8个小时,即2005年7月30日15点18分20秒。下面是个例子:


#i nclude "time.h"
#i nclude "stdio.h"
int main(void)
{
struct tm *local;
time_t t;
t=time(NULL);
local=localtime(&t);
printf("Local hour is: %d\n",local->tm_hour);
local=gmtime(&t);
printf("UTC hour is: %d\n",local->tm_hour);
return 0;
}


运行结果是:


Local hour is: 15
UTC hour is: 7


4.3 固定的时间格式


我们可以通过asctime()函数和ctime()函数将时间以固定的格式显示出来,两者的返回值都是char*型的字符串。返回的时间格式为:


星期几 月份 日期 时:分:秒 年\n\0
例如:Wed Jan 02 02:03:55 1980\n\0


其中\n是一个换行符,\0是一个空字符,表示字符串结束。下面是两个函数的原型:


char * asctime(const struct tm * timeptr);
char * ctime(const time_t *timer);


其中asctime()函数是通过tm结构来生成具有固定格式的保存时间信息的字符串,而ctime()是通过日历时间来生成时间字符串。这样的话,asctime()函数只是把tm结构对象中的各个域填到时间字符串的相应位置就行了,而ctime()函数需要先参照本地的时间设置,把日历时间转化为本地时间,然后再生成格式化后的字符串。在下面,如果lt是一个非空的time_t变量的话,那么:


printf(ctime(<));


等价于:


struct tm *ptr;
ptr=localtime(<);
printf(asctime(ptr));


那么,下面这个程序的两条printf语句输出的结果就是不同的了(除非你将本地时区设为世界标准时间所在的时区):


#i nclude "time.h"
#i nclude "stdio.h"
int main(void)
{
struct tm *ptr;
time_t lt;
lt =time(NULL);
ptr=gmtime(<);
printf(asctime(ptr));
printf(ctime(<));
return 0;
}


运行结果:


Sat Jul 30 08:43:03 2005
Sat Jul 30 16:43:03 2005


4.4 自定义时间格式


我们可以使用strftime()函数将时间格式化为我们想要的格式。它的原型如下:


size_t strftime(
    char *strDest,
    size_t maxsize,
    const char *format,
    const struct tm *timeptr 
);


我们可以根据format指向字符串中格式命令把timeptr中保存的时间信息放在strDest指向的字符串中,最多向strDest中存放maxsize个字符。该函数返回向strDest指向的字符串中放置的字符数。


函数strftime()的操作有些类似于sprintf():识别以百分号(%)开始的格式命令集合,格式化输出结果放在一个字符串中。格式化命令说明串strDest中各种日期和时间信息的确切表示方法。格式串中的其他字符原样放进串中。格式命令列在下面,它们是区分大小写的。


%a 星期几的简写 
%A 星期几的全称 
%b 月分的简写 
%B 月份的全称 
%c 标准的日期的时间串 
%C 年份的后两位数字 
%d 十进制表示的每月的第几天 
%D 月/天/年 
%e 在两字符域中,十进制表示的每月的第几天 
%F 年-月-日 
%g 年份的后两位数字,使用基于周的年 
%G 年分,使用基于周的年 
%h 简写的月份名 
%H 24小时制的小时 
%I 12小时制的小时
%j 十进制表示的每年的第几天 
%m 十进制表示的月份 
%M 十时制表示的分钟数 
%n 新行符 
%p 本地的AM或PM的等价显示 
%r 12小时的时间 
%R 显示小时和分钟:hh:mm 
%S 十进制的秒数 
%t 水平制表符 
%T 显示时分秒:hh:mm:ss 
%u 每周的第几天,星期一为第一天 (值从0到6,星期一为0)
%U 第年的第几周,把星期日做为第一天(值从0到53)
%V 每年的第几周,使用基于周的年 
%w 十进制表示的星期几(值从0到6,星期天为0)
%W 每年的第几周,把星期一做为第一天(值从0到53) 
%x 标准的日期串 
%X 标准的时间串 
%y 不带世纪的十进制年份(值从0到99)
%Y 带世纪部分的十制年份 
%z,%Z 时区名称,如果不能得到时区名称则返回空字符。
%% 百分号


如果想显示现在是几点了,并以12小时制显示,就象下面这段程序:


#i nclude “time.h”
#i nclude “stdio.h”
int main(void)
{
struct tm *ptr;
time_t lt;
char str[80];
lt=time(NULL);
ptr=localtime(<);
strftime(str,100,"It is now %I %p",ptr);
printf(str);
return 0;
}


其运行结果为:
It is now 4PM


而下面的程序则显示当前的完整日期:


#i nclude 
#i nclude 


void main( void )
{
         struct tm *newtime;
         char tmpbuf[128];
         time_t lt1;
         time( <1 );
         newtime=localtime(<1);
         strftime( tmpbuf, 128, "Today is %A, day %d of %B in the year %Y.\n", newtime);
         printf(tmpbuf);
}


运行结果:


Today is Saturday, day 30 of July in the year 2005.


4.5 计算持续的时间长度


有时候在实际应用中要计算一个事件持续的时间长度,比如计算打字速度。在第1节计时部分中,我已经用clock函数举了一个例子。Clock()函数可以精确到毫秒级。同时,我们也可以使用difftime()函数,但它只能精确到秒。该函数的定义如下:


double difftime(time_t time1, time_t time0);


虽然该函数返回的以秒计算的时间间隔是double类型的,但这并不说明该时间具有同double一样的精确度,这是由它的参数觉得的(time_t是以秒为单位计算的)。比如下面一段程序:


#i nclude “time.h”
#i nclude “stdio.h”
#i nclude “stdlib.h”
int main(void)
{
time_t start,end;
start = time(NULL);
system("pause");
end = time(NULL);
printf("The pause used %f seconds.\n",difftime(end,start));//<-
system("pause");
return 0;
}


运行结果为:
请按任意键继续. . .
The pause used 2.000000 seconds.
请按任意键继续. . .


可以想像,暂停的时间并不那么巧是整整2秒钟。其实,你将上面程序的带有“//<-”注释的一行用下面的一行代码替换:


printf("The pause used %f seconds.\n",end-start);


其运行结果是一样的。


4.6 分解时间转化为日历时间


这里说的分解时间就是以年、月、日、时、分、秒等分量保存的时间结构,在C/C++中是tm结构。我们可以使用mktime()函数将用tm结构表示的时间转化为日历时间。其函数原型如下:


time_t mktime(struct tm * timeptr);


其返回值就是转化后的日历时间。这样我们就可以先制定一个分解时间,然后对这个时间进行操作了,下面的例子可以计算出1997年7月1日是星期几:


#i nclude "time.h"
#i nclude "stdio.h"
#i nclude "stdlib.h"
int main(void)
{
struct tm t;
time_t t_of_day;
t.tm_year=1997-1900;
t.tm_mon=6;
t.tm_mday=1;
t.tm_hour=0;
t.tm_min=0;
t.tm_sec=1;
t.tm_isdst=0;
t_of_day=mktime(&t);
printf(ctime(&t_of_day));
return 0;
}


运行结果:


Tue Jul 01 00:00:01 1997


现在注意了,有了mktime()函数,是不是我们可以操作现在之前的任何时间呢?你可以通过这种办法算出1945年8月15号是星期几吗?答案是否定的。因为这个时间在1970年1月1日之前,所以在大多数编译器中,这样的程序虽然可以编译通过,但运行时会异常终止。


5.总结


本文介绍了标准C/C++中的有关日期和时间的概念,并通过各种实例讲述了这些函数和数据结构的使用方法。笔者认为,和时间相关的一些概念是相当重要的,理解这些概念是理解各种时间格式的转换的基础,更是应用这些函数和数据结构的基础。




标准C++ 时间日期函数 
一、MFC类
MFC提供了两个日期和时间类CTime和CTimeSpan,分别代表相对时间和绝对时间。CTime是基于格林威治平均时间(GMT)的,本地的时间由环境变量TZ决定。CTimeSpan代表了时间间隔。 


CTime类由下列成员函数: 


CTime() 


创建一个CTime对象。 


GetCurrentTime() 


由当前时间创建一个CTime对象。 


GetTime() 


由CTime对象返回一个time_t变量。 


GetYear() 


获取CTime对象代表的年。 


GetMonth() 


获取CTime对象代表的月。 


GetDay() 获取CTime对象代表的日期。 


GetHour() 获取CTime对象代表的小时。 


GetMinute()获取CTime对象代表的分。 


GetSecond() 获取CTime对象代表的秒。 


GetDayOfWeek() 获取CTime对象代表的周日,1代表周日,2代表周-等等。 


Format() 将字符串转换成一个基于本地时区的格式字符串。 


FormatGmt() 将字符串转换成一个基于UTC(世界时)的格式字符串。 


operator = 赋予新的时间。 


operator + 增加CTime和CTimeSpan对象。 


operator – 减小CTime和CTimeSpan对象。 


operator += CTime对象加一个CTimeSpan对象。 


operator -= CTime对象减一个CTimeSpan对象。 


operator == 比较两个绝对时间是否相等。 


operator != 比较两个绝对时间是否不相等。 


operator < 比较两个绝对时间,是否前一个大于后一个。 


operator > 比较两个绝对时间,是否前一个小于后一个。 


operator >= 比较两个绝对时间,是否前一个大于等于后一个。 


operator <= 比较两个绝对时间,是否前一个小于等于后一个 


Include head file time.h, though it's a C include file, C++ certainly can use it. 
Under C++, you can include <ctime> instead of <time.h> 
_____________________________________________________ 


time.h 


@函数名称: localtime 
函数原型: struct tm *localtime(const time_t *timer) 
函数功能: 返回一个以tm结构表达的机器时间信息 
函数返回: 以tm结构表达的时间,结构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 
int tm_yday; //记录今天是一年里面的第几天,从1月1日起,0-365
int tm_isdst; //日光节约时间旗标,大于0表示作用中,等于0表示不作用
}; 
参数说明: timer-使用time()函数获得的机器时间 
所属文件: <time.h> 


#include <time.h> 
#include <stdio.h> 
#include <dos.h> 
int main() 

time_t timer; 
struct tm *tblock; 
timer=time(NULL); 
tblock=localtime(&timer); 
printf("Local time is: %s",asctime(tblock)); 
return 0; 



@函数名称: asctime 
函数原型: char* asctime(struct tm * ptr) 
函数功能: 得到机器时间(日期时间转换为ASCII码) 
函数返回: 返回的时间字符串格式为:星期,月,日,小时:分:秒,年 
参数说明: 结构指针ptr应通过函数localtime()和gmtime()得到 
所属文件: <time.h> 


#include <stdio.h> 
#include <string.h> 
#include <time.h> 
int main() 

struct tm t; 
char str[80]; 
t.tm_sec=1; 
t.tm_min=3; 
t.tm_hour=7; 
t.tm_mday=22; 
t.tm_mon=11; 
t.tm_year=56; 
t.tm_wday=4; 
t.tm_yday=0; 
t.tm_isdst=0; 
strcpy(str,asctime(&t)); 
printf("%s",str); 
return 0; 



@函数名称: ctime 
函数原型: char *ctime(long time) 
函数功能: 得到日历时间 
函数返回: 返回字符串格式:星期,月,日,小时:分:秒,年 
参数说明: time-该参数应由函数time获得 
所属文件: <time.h> 


#include <stdio.h> 
#include <time.h> 
int main() 

time_t t; 
time(&t); 
printf("Today's date and time: %s",ctime(&t)); 
return 0; 



@函数名称: difftime 
函数原型: double difftime(time_t time2, time_t time1) 
函数功能: 得到两次机器时间差,单位为秒 
函数返回: 时间差,单位为秒 
参数说明: time1-机器时间一,time2-机器时间二.该参数应使用time函数获得 
所属文件: <time.h> 


#include <time.h> 
#include <stdio.h> 
#include <dos.h> 
#include <conio.h> 
int main() 

time_t first, second; 
clrscr(); 
first=time(NULL); 
delay(2000); 
second=time(NULL); 
printf("The difference is: %f seconds",difftime(second,first)); 
getch(); 
return 0; 



@函数名称: gmtime 
函数原型: struct tm *gmtime(time_t *time) 
函数功能: 得到以结构tm表示的时间信息 
函数返回: 以结构tm表示的时间信息指针 
参数说明: time-用函数time()得到的时间信息 
所属文件: <time.h> 


#include <stdio.h> 
#include <stdlib.h> 
#include <time.h> 
#include <dos.h> 
char *tzstr="TZ=PST8PDT"; 
int main() 

time_t t; 
struct tm *gmt, *area; 
putenv(tzstr); 
tzset(); 
t=time(NULL); 
area=localtime(&t); 
printf("Local time is:%s", asctime(area)); 
gmt=gmtime(&t); 
printf("GMT is:%s", asctime(gmt)); 
return 0; 



@函数名称: time 
函数原型: time_t time(time_t *timer) 
函数功能: 得到机器的日历时间或者设置日历时间 
函数返回: 机器日历时间 
参数说明: timer=NULL时得到机器日历时间,timer=时间数值时,用于设置日历时间,time_t是一个long类型 
所属文件: <time.h> 


#include <time.h> 
#include <stdio.h> 
#include <dos.h> 
int main() 

time_t t; 
t=time(); 
printf("The number of seconds since January 1,1970 is %ld",t); 
return 0; 



@函数名称: tzset 
函数原型: void tzset(void) 
函数功能: UNIX兼容函数,用于得到时区,在DOS环境下无用途 
函数返回: 
参数说明: 
所属文件: <time.h> 


#include <time.h> 
#include <stdlib.h> 
#include <stdio.h> 
int main() 

time_t td; 
putenv("TZ=PST8PDT"); 
tzset(); 
time(&td); 
printf("Current time=%s",asctime(localtime(&td))); 
return 0; 



/方案— 优点:仅使用C标准库;缺点:只能精确到秒级
#include <time.h>
#include <stdio.h>
int main( void )
{
time_t t = time(0);
char tmp[64];
strftime( tmp, sizeof(tmp), "%Y/%m/%d %X %A 本年第%j天 %z",localtime(&t) );
puts( tmp );
return 0;
}
size_t strftime(char *strDest, size_t maxsize, const char *format, const struct tm *timeptr);
根据格式字符串生成字符串。
struct tm *localtime(const time_t *timer);
取得当地时间,localtime获取的结果由结构tm返回
返回的字符串可以依下列的格式而定:
%a 星期几的缩写。Eg:Tue
%A 星期几的全名。 Eg: Tuesday
%b 月份名称的缩写。
%B 月份名称的全名。
%c 本地端日期时间较佳表示字符串。
%d 用数字表示本月的第几天 (范围为 00 至 31)。日期
%H 用 24 小时制数字表示小时数 (范围为 00 至 23)。
%I 用 12 小时制数字表示小时数 (范围为 01 至 12)。
%j 以数字表示当年度的第几天 (范围为 001 至 366)。
%m 月份的数字 (范围由 1 至 12)。
%M 分钟。
%p 以 ''AM'' 或 ''PM'' 表示本地端时间。
%S 秒数。
%U 数字表示为本年度的第几周,第一个星期由第一个周日开始。
%W 数字表示为本年度的第几周,第一个星期由第一个周一开始。
%w 用数字表示本周的第几天 ( 0 为周日)。
%x 不含时间的日期表示法。
%X 不含日期的时间表示法。 Eg: 15:26:30
%y 二位数字表示年份 (范围由 00 至 99)。
%Y 完整的年份数字表示,即四位数。 Eg:2008
%Z(%z) 时区或名称缩写。Eg:中国标准时间
%% % 字符。 


//方案二 优点:能精确到毫秒级;缺点:使用了windows API
#include <windows.h>
#include <stdio.h>
int main( void )
{
SYSTEMTIME sys;
GetLocalTime( &sys );
printf( "%4d/%02d/%02d %02d:%02d:%02d.%03d 星期%1d\n",sys.wYear,sys.wMonth,sys.wDay,sys.wHour,sys.wMinute, sys.wSecond,sys.wMilliseconds,sys.wDayOfWeek);
return 0;



//方案三,优点:利用系统函数,还能修改系统时间
//此文件必须是c++文件
#include<stdlib.h>
#include<iostream>
using namespace std;
void main()
{
system("time");



//方案四,将当前时间折算为秒级,再通过相应的时间换算即可
//此文件必须是c++文件
#include<iostream>
#include<ctime>
using namespace std;
int main()
{
time_t now_time;
now_time = time(NULL);
cout<<now_time;
return 0;



上次实验的时候调用的GETSYSYTEMTIME居然显示的是系统开机时间又不是做定时炸弹最后只有换GETLOCALTIME了都聚集在一起,以后好用~~

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值