FILETIME, SYSTEMTIME and time_t

本文深入解析了多种时间类型,包括UTC、日历时间、FILETIME、SYSTEMTIME、tm和time_t的概念及其相互间的转换方法。重点介绍了不同时间类型在Windows API中的应用。


先说个概念: 
Coordinated Universal Time(UTC):又称为世界标准时间;
比如,中国内地的时间与UTC的时差为+8,也就是UTC+8。 
Calendar Time:日历时间,是用“从一个标准时间点到此时的时间经过的秒数”来表示的时间。日历时间对于不同的编译器,是不同的。
但是无论你在哪一个时区,在同一时刻对同一个标准时间点来说,日历时间都是一样的。

例如,time_t就是一种日历时间,其值表示从UTC(Coordinated Universal Time)时间1970年1月1日00:00:00(称为UNIX系统的Epoch时间) 到当前时刻的秒数。


这里面涉及了四个类型  :

*1.FILETIME:结构持有的64位无符号的文件的日期和时间值。此值表示自UTC(Coordinated Universal Time)时间1601年1月1日开始的100纳秒为单位的时间。
*2.SYSTEMTIME:这个和tm有点类似,代表系统时间。其中,SYSTEMTIME.year的值为多少就表示公元哪一年。
*3.tm:Systemtime-1900得到tm.year,其他的还有些差别 tm.year的值加上1900才表示公元哪一年。
*4.time_t:是日历时间;其值表示从UTC(Coordinated Universal Time)时间1970年1月1日00:00:00(称为UNIX系统的Epoch时间) 到当前时刻的秒数。


FileTime转换为Time_t


//#### FileTimeToTime_t ####
void  FileTimeToTime_t(  FILETIME  ft,  time_t  *t  )  
{  
 LONGLONG  ll;  
 
 ULARGE_INTEGER            ui;  
 ui.LowPart            =  ft.dwLowDateTime;  
 ui.HighPart            =  ft.dwHighDateTime;  
 
 ll            =  ft.dwHighDateTime  <<  32  +  ft.dwLowDateTime;  //这一步是不是多余的
 
 *t            =  ((LONGLONG)(ui.QuadPart  -  116444736000000000)  /  10000000); 

//将ui.QuadPart的数据类型ULONGLONG强制转换为time_tLONGLONG数据类型

//


struct {
DWORD LowPart;
DWORD HighPart;
};  和ULONGLONG QuadPart;组成共用体ULARGE_INTEGER

注释:公元1601年到公元1970年相差369年,转化为秒数即为11644473600秒。

参见:

Windows API的时间结构体、时间转换及时间获取

如何在C++中将filetime时间转化为字符串?


//********************************************************************/

或是如下方法:

如果编译器中的long类型为4字节时,则 time_t(typedef time_t __int64)

filetime(filetime结构体中有两个unsigned long类型的变量)占用字节大小一样(__int64是有符号吧),则filetime先可以转化为time_t,再由time_t转化为tm:



filetime st;


__int64 d = *(__int64 *)&st;

//因为filetime结构体中的两个变量是unsigned long类型的,而*(__int64 *)&st只是表示将unsigned long类型的变量st的数据按照__int64(=longlong)来解释读取而已,并未实际将unsigned long类型的变量st的数据转换成__int64(=longlong)数据类型格式的。

__int64 d *(__int64 *)&st;应该改为__int64 d =(__int64)st;不行,因为编译器一般只存在从基本数据类型到基本数据类型的类型转换函数,而像从复合数据类型到基本数据类型等的类型转换函数一般不存在,即编译器可能不存在从filetime结构体类型到__int64的类型转换函数。


参见:
 

强制类型转换和隐式类型转换|__int64 d = *(__int64 *)&st;

中的

为什么filetime数据类型的变量st可以通过如下方式正确读取其所要表示的数值:

__int64 d = *(__int64 *)&st?


// or

//__int64 d2;
//memcpy(
& d2, & st, sizeof (d2));

(d -  116444736000000000)  /  10000000

注释:

CRT中的时间(time_t和tm)

 

Windows中的时间(SYSTEMTIME和FILETIME)

上一篇中介绍了C运行库中的时间处理函数。这一篇介绍Windows SDk中提供的时间函数。两种时间系统之间没有本质区别(事实上CRT时间是用Windows时间实现的,当然这是说的VC实现),同样提供本地时间和UTC时间之间的转换。不过, CRT中的tm时间在SDK中对应为系统时间(SYSTEMTIME),CRT中的time_t时间在SDK中对应的为文件时间(FILETIME), 那个"特殊时刻"也变成1601年1月1日的子夜。

参见:

FILETIME SYSTEMTIME tm time_t

CRT中的时间(time_t和tm)

 

Windows中的时间(SYSTEMTIME和FILETIME)

FILETIME, SYSTEMTIME 与 time_t 相互转换

filetime 转化为 time_t 百度

filetime 时间单位 百度

time_t 时间单位 百度

附加:


补习一下时间单位(http://blog.vckbase.com/zaboli/archive/2005/06/29/8969.aspx)

1ms (毫秒)1毫秒=0.001秒=10-3秒(millisecond)
1μs (微秒)1微秒=0.000001=10-6秒(microsecond)
1ns (纳秒)1纳秒=0.000000001秒=10-9秒(nanosecond)
1ps (皮秒)1皮秒=0.000000000001秒=10-12秒
1fs (飞秒)1飞秒=0.00000000000001秒=10-15秒

FILETIME

1功能

FILETIME结构持有的64位无符号的文件的日期和时间值。此值表示自1601年1月1日开始的100纳秒为单位的时间。

2语法

typedef struct _FILETIME {
DWORD dwLowDateTime; /* 低 32 位 */
DWORD dwHighDateTime; /* 高 32 位 */
} FILETIME, *PFILETIME, *LPFILETIME;

3成员

dwLowDateTime
低位32位的文件的时间值。
dwHighDateTime
高位32位的文件的时间值。

4要求

头文件: WINDEF.H [2]



FILETIME是一个__int64。可以这样转化成__int64
__int64 d  =   * (__int64  * ) & st;

// or

__int64 d2;
memcpy(
& d2, & st, sizeof (d2));

转化为__int64是一个以100纳秒为单位的值

参考:

Win32时间类型FILETIME/SYSTEMTIME/WINDOWSTIME(FILETIME这个名字很奇怪,其实他跟FILE并没有直接关系,只是很多File的API中,都以这个为时间的类型)


FILETIME, SYSTEMTIME 与 time_t 相互转换  


//************************************************************
//FILETIME, SYSTEMTIME 与 time_t 相互转换 

//#####SYSTEMTIME 与 FILETIME相互转换##### 
//可以使用系统函数
//FileTimeToSystemTime(&ftcreate,&stcreate);  

//参数:
//(lpFileTime As FILETIME, lpSystemTime As SYSTEMTIME) 
//说明 
//根据一个FILETIME结构的内容,装载一个SYSTEMTIME结构 
//返回值 
//Long,非零表示成功,零表示失败。会设置GetLastError 
//参数表 
//参数 类型及说明 
//lpFileTime FILETIME,包含了文件时间的一个结构 
//lpSystemTime SYSTEMTIME,用于装载系统时间信息的一个结构

//#####SYSTEMTIME 与 time_t相互转换#####

//#### Time_tToSystemTime ####
void TimetToSystemTime( time_t t, LPSYSTEMTIME pst )
{
 FILETIME ft; 
 LONGLONG ll = Int32x32To64(t, 10000000) + 116444736000000000;
 ft.dwLowDateTime = (DWORD) ll;
 ft.dwHighDateTime = (DWORD)(ll >> 32);
 
 FileTimeToSystemTime( &ft, pst );
}

//#### SystemTimeToTime_t ####
void SystemTimeToTime_t( SYSTEMTIME st, time_t *pt )
{
 FILETIME ft;
 SystemTimeToFileTime( &st, &ft );
 
 LONGLONG ll;
 
 ULARGE_INTEGER ui;
 ui.LowPart = ft.dwLowDateTime;
 ui.HighPart = ft.dwHighDateTime;
 
 ll = (ft.dwHighDateTime << 32) + ft.dwLowDateTime;
 
 *pt = (DWORD)((LONGLONG)(ui.QuadPart - 116444736000000000) / 10000000);
}

//#### FileTimeToTime_t ####
void  FileTimeToTime_t(  FILETIME  ft,  time_t  *t  )  
{  
 LONGLONG  ll;  
 
 ULARGE_INTEGER            ui;  
 ui.LowPart            =  ft.dwLowDateTime;  
 ui.HighPart            =  ft.dwHighDateTime;  
 
 ll            =  ft.dwHighDateTime  <<  32  +  ft.dwLowDateTime;  //这一步是不是多余的
 
 *t            =  ((LONGLONG)(ui.QuadPart  -  116444736000000000)  /  10000000); 

//将ui.QuadPart的数据类型ULONGLONG强制转换为time_tLONGLONG数据类型


}  

//


struct {
DWORD LowPart;
DWORD HighPart;
};  和ULONGLONG QuadPart;组成共用体ULARGE_INTEGER

//********************************************************************/

参见:

这一篇介绍Windows SDk中提供的时间函数。两种时间系统之间没有本质区别(事实上CRT时间是用Windows时间实现的,当然这是说的VC实现),同样提供本地时间和UTC时间之间的转换。不过CRT中的tm时间在SDK中对应为系统时间(SYSTEMTIME),CRT中的time_t时间在SDK中对应的为文件时间(FILETIME),那个"特殊时刻"也变成1601年1月1日的子夜。

    当然,首先要弄清楚FILETIME与SYSTEMTIME定义。

   

[cpp]  view plain copy
  1. typedef struct _FILETIME {  
  2.                           DWORD dwLowDateTime;  
  3.                           DWORD dwHighDateTime;  
  4. } FILETIME, *PFILETIME;  
  5. typedef struct _SYSTEMTIME {  
  6.                             WORD wYear;  
  7.                             WORD wMonth;  
  8.                             WORD wDayOfWeek;  
  9.                             WORD wDay;  
  10.                             WORD wHour;  
  11.                             WORD wMinute;  
  12.                             WORD wSecond;  
  13.                             WORD wMilliseconds;  
  14. } SYSTEMTIME, *PSYSTEMTIME;  

    比较一下,很明显,FILETIME与time_t类似,是64位整型,不过FILETIME是以100纳秒(ns)为单位。SYSTEMTIME与tm类似,不过多了一项wMilliseconds。可以看出,SDK时间比CRT的时间提供了更高的精度。同时SDK提供了更丰富的函数来处理时间。

   

[c-sharp]  view plain copy
  1. void GetSystemTime(  
  2.                    LPSYSTEMTIME lpSystemTime);  
  3. void GetLocalTime(  
  4.                   LPSYSTEMTIME lpSystemTime);  

    这两个函数获得SYSTEMTIME形式的当前时间,不过GetSystemTime函数获得当前的UTC时间,GetLocalTime获得当前的本地时间,可以想象,获得的两个时间存在着时差。类似于CRT中提供tm与time_t之间的转换,SDK也提供了两个函数来转换SYSTEMTIME时间与FILETIME时间。

   

[cpp]  view plain copy
  1. BOOL SystemTimeToFileTime(  
  2.                           const SYSTEMTIME* lpSystemTime,  
  3.                           LPFILETIME lpFileTime);  
  4. BOOL FileTimeToSystemTime(  
  5.                           const FILETIME* lpFileTime,  
  6.                           LPSYSTEMTIME lpSystemTime);  

    函数命名很self-explanatory,就不用多说了吧。

    SDK还提供了两个很有趣的函数。

   

[cpp]  view plain copy
  1. BOOL LocalFileTimeToFileTime(  
  2.                              const FILETIME* lpLocalFileTime,  
  3.                              LPFILETIME lpFileTime);  
  4. BOOL FileTimeToLocalFileTime(  
  5.                              const FILETIME* lpFileTime,  
  6.                              LPFILETIME lpLocalFileTime);  

    LocalFileTimeToFileTime函数将本地的FILETIME时间转换为对应的UTC的FILETIME时间。我觉得,这个函数只是通过将本地时间减去与UTC时间的时间差来实现转换,比如在东八区的本地时间转换为对应的UTC时间,只需要将本地时间减去8*60*60*1000*1000*10(单位100ns)。类似,FileTimeToLocalFileTime函数是将UTC时间转换为本地时间,它只是将减去时间差换成加上时间差。

    了解了这些功能,让我们用代码说话吧。

   

[cpp]  view plain copy
  1. #include <stdlib.h>  
  2. #include <stdio.h>  
  3. #include <time.h>  
  4. #include <windows.h>  
  5.   
  6.   
  7. int main()  
  8. {  
  9.     SYSTEMTIME      stLocal, stUTC, stUTC2;  
  10.     FILETIME        ftLocal, ftUTC, ft;  
  11.     ULARGE_INTEGER  uli;  
  12.   
  13.     GetLocalTime(&stLocal);  
  14.     GetSystemTime(&stUTC);  
  15.     printf("Local System Time(YYYY-MM-DD HH:MM:SS): %d-%d-%d %d:%d:%d/n", stLocal.wYear, stLocal.wMonth,  
  16.         stLocal.wDay, stLocal.wHour, stLocal.wMinute, stLocal.wSecond);  
  17.     printf("UTC System Time  (YYYY-MM-DD HH:MM:SS): %d-%d-%d %d:%d:%d/n", stUTC.wYear, stUTC.wMonth,  
  18.         stUTC.wDay, stUTC.wHour, stUTC.wMinute, stUTC.wSecond);  
  19.   
  20.     SystemTimeToFileTime(&stLocal, &ftLocal);  
  21.     uli.LowPart = ftLocal.dwLowDateTime;  
  22.     uli.HighPart = ftLocal.dwHighDateTime;  
  23.     printf("Local File Time: %llu/n", uli.QuadPart);  
  24.   
  25.     LocalFileTimeToFileTime(&ftLocal, &ftUTC);  
  26.     uli.LowPart = ftUTC.dwLowDateTime;  
  27.     uli.HighPart = ftUTC.dwHighDateTime;  
  28.     printf("UTC File Time: %llu/n", uli.QuadPart);  
  29.   
  30.     FileTimeToSystemTime(&ftUTC, &stUTC2);  
  31.     printf("UTC System Time2 (YYYY-MM-DD HH:MM:SS): %d-%d-%d %d:%d:%d/n", stUTC2.wYear, stUTC2.wMonth,  
  32.         stUTC2.wDay, stUTC2.wHour, stUTC2.wMinute, stUTC2.wSecond);  
  33.   
  34.     return EXIT_SUCCESS;  
  35. }  

    程序输出结果如下:

    程序输出结果

    代码13行GetLocalTime函数获得当前的本地SYSTEMTIME时间,14行获得对应的UTC的SYSTEMTIME时间,如输出结果前两行所显示,两者相差8小时(凌晨还在写博客,表扬下自己。。。)。

    20行SystemTimeToFileTime函数将本地SYSTEMTIME时间转换为方便计算的本地FILETIME形式时间,如输出结果第三行所显示。

    25行LocalFileTimeToFileTime函数将本地FileTime时间转换为对应的UTC的FILETIME时间,如输出结果第四行所显示。就像前面介绍的,如果你将输出结果第三,四两行所显示的数字相减,并除以10*1000*1000*60*60,你将会得出8,你可以算下试试,记住FILETIME是以100纳秒为单位的。

    最后30行FileTimeToSystemTime将FILETIME时间转换为SYSTEMTIME时间。可以看出输出结果中第五行与第二行相同,这是必须的,因为两者都是当前本地时间对应的UTC时间。


CRT中的时间(time_t和tm)

分类: C/C++ General Programming   4147人阅读  评论(3)  收藏  举报

    时间处理时实际项目中经常碰到的问题,这里介绍最常用的时间处理函数。

    首先介绍基本的时间概念。时间一般分为两种,一种是本地时间(Local Time),一种是协调世界时间(Coordinated Universal Time ,UTC),也就是传说中的格林威治时间。本地时间与UTC时间之间的差即为时差,比如,北京时间(东八区)比UTC时间晚8个小时。

    C运行库中处理时间的函数主要是这四个:

    

[cpp]  view plain copy
  1. time_t time(  
  2.             time_t *timer);  
  3.    

    time_t类型为32位或64位整型,具体类型由编译系统决定。此函数用来获得从1970年1月1日子夜(这个时刻在不同的CRT实现中可能会不一样)到当前时刻以来所流逝的时间,以秒为单位。这个时间差叫做日历时间(Calendar Time )。

    这是当然让我困惑的地方:这个特殊的时刻——1970年1月1日零时零分零秒——是指本地时间呢,还是UTC时间呢?我认为是本地时间,也就是各个时区自己的1970年1月1日零时零分零秒。可以设想这样一种情况,如果全球24时区各有一台电脑,都依次在自己所在时区的本地时间1970年1月1日零时1分零秒调用time函数,那么返回值都是60。注意,这里是依次调用(事实上是每隔1小时),而不是想象中的同时调用,这是因为相邻时区的同一本地时间,总是相差1小时。

    当然,time_t型的时间方便计算机处理,但普通用户无法理解这种数字。所以我们通常需要将time_t型时间转换成我们平常所见的年月日形式。CRT中为此定义了tm结构。

   

[cpp]  view plain copy
  1. struct tm {  
  2.     int tm_sec;     /* seconds after the minute - [0,59] */  
  3.     int tm_min;     /* minutes after the hour - [0,59] */  
  4.     int tm_hour;    /* hours since midnight - [0,23] */  
  5.     int tm_mday;    /* day of the month - [1,31] */  
  6.     int tm_mon;     /* months since January - [0,11] */  
  7.     int tm_year;    /* years since 1900 */  
  8.     int tm_wday;    /* days since Sunday - [0,6] */  
  9.     int tm_yday;    /* days since January 1 - [0,365] */  
  10.     int tm_isdst;   /* daylight savings time flag */  
  11. };  

    注释中已详细解释了各个字段的用法。显然这个结构中的字段对用户更有意义。我们通常用localtime_s函数将time_t时间转换为tm时间。

   

[cpp]  view plain copy
  1. errno_t localtime_s(  
  2.                     struct tm* _tm,  
  3.                     const time_t *time);  

    其中第二个参数为传入的time_t时间,第一个参数为返回的tm时间。由函数名可看出,返回的tm时间表示的是本地时间。当然,我们有时候也需要获得对应的UTC时间,这时我们需要gmtime函数。

   

[cpp]  view plain copy
  1. errno_t gmtime_s(  
  2.                  struct tm* _tm,  
  3.                  const time_t* time);  

    后面我们会看到两者的区别。

    我们知道了如何将time_t时间转换为tm时间。同样,我们会需要将tm表示的时间转换为time_t时间。这时我们需要mktime函数。

   

[cpp]  view plain copy
  1. time_t mktime(  
  2.               struct tm *timeptr);  

    此函数返回从"特殊时刻"到参数表示的时刻之间流逝的日历时间。另外还有个很好用的特性,就是它能修正传进来的tm结构中各字段的取值范围。比如,如果你将tm.tm_mon设为1,tm.tm_day设为33,然后以其为参数调用mktime函数,此函数会将tm.tm_mon修正为2,tm.tm_day修正为2。具体用法参照MSDN。

    我们来分析下面示例代码:

   

[cpp]  view plain copy
  1. #include <stdlib.h>  
  2. #include <stdio.h>  
  3. #include <time.h>  
  4.   
  5.   
  6. int main()  
  7. {  
  8.     struct tm   tmLocal, tmUTC;  
  9.     time_t  tNow;  
  10.     //Get current calendar time  
  11.     time(&tNow);  
  12.     printf("Time Now from time(): %llu/n", tNow);     
  13.     //Get current local time  
  14.     localtime_s(&tmLocal, &tNow);   
  15.     printf("Local Time(YYYY-MM-DD HH:MM:SS): %d-%d-%d %d:%d:%d/n", tmLocal.tm_year + 1900, tmLocal.tm_mon,  
  16.         tmLocal.tm_mday, tmLocal.tm_hour, tmLocal.tm_min, tmLocal.tm_sec);  
  17.     //Get UTC time corresponding to current local time, and tmLocal.tm_hour - tmUTC.tm_hour = 8  
  18.     gmtime_s(&tmUTC, &tNow);      
  19.     printf("UTC Time  (YYYY-MM-DD HH:MM:SS): %d-%d-%d %d:%d:%d/n", tmUTC.tm_year + 1900, tmUTC.tm_mon,  
  20.         tmUTC.tm_mday, tmUTC.tm_hour, tmUTC.tm_min, tmUTC.tm_sec);  
  21.     //convert tmLocal to calendar time  
  22.     tNow = mktime(&tmLocal);      
  23.     printf("Time Now from mktime(): %llu/n", tNow);  
  24.   
  25.     return EXIT_SUCCESS;  
  26. }  

    输出结果如下:

    输出结果

    上面代码中,11行time函数获得从"特殊时刻"到当前时刻的日历时间,如输出结果中的第一行显示的1267192581秒。

    14行localtime_s函数将日历时间转换为本地tm时间,如输出结果第二行。

    18行gmtime_s函数将将日历时间转换为对应的UTC的tm时间,如输出结果第三行显示。很容易看出,第二,三行输出的时间相差8小时,因为我在东八区。如果你修改自己电脑的时区(在控制面板的Date and Time中修改),再运行此程序,比较两次的运行结果,你就可以更好的理解了。

    22行mktime函数将tm时间转换为日历时间,输出结果中第四行显示的结果与第一行一样,这是必须的。。。

FILETIME SYSTEMTIME tm time_t


TimeConvert.h

[cpp]  view plain copy
  1. // TimeConvert.h: interface for the TimeConvert class.  
  2. //  
  3. //@anthor:TT  
  4. //@time:2012.11.22  
  5. //@introduction:今天由于处理一个时间,搞得焦头烂额,于是乎打算封装一个转换类,方便自己以后使用,  
  6. /*先说个概念: 
  7. Coordinated Universal Time(UTC):又称为世界标准时间;时间1970年1月1日00:00:00(称为UNIX系统的Epoch时间)到当前时刻的秒数。 
  8. 比如,中国内地的时间与UTC的时差为+8,也就是UTC+8。 
  9. Calendar Time:日历时间,是用“从一个标准时间点到此时的时间经过的秒数”来表示的时间。对于不同的编译器,是不同的。 
  10. 但是无论你在哪一个时区,在同一时刻对同一个标准时间点来说,日历时间都是一样的。 
  11. */  
  12. //这里面涉及了三个类型  
  13. /* 
  14. *1.FILETIME:结构持有的64位无符号的文件的日期和时间值。此值表示自1601年1月1日开始的100纳秒为单位的时间。 
  15. *2.SYSTEMTIME:这个和tm有点类似,代表系统时间 
  16. *3.tm:Systemtime-1900得到tm.year,其他的还有些差别 
  17. *4.time_t:是日历时间;其值表示从CUT(Coordinated Universal Time)时间1970年1月1日00:00:00(称为UNIX系统的Epoch时间) 
  18. *到当前时刻的秒数。 
  19. */  
  20. //////////////////////////////////////////////////////////////////////  
  21.   
  22. #if !defined(AFX_TIMECONVERT_H__87F35906_DBD6_46F1_BF78_2FA34FCB8409__INCLUDED_)  
  23. #define AFX_TIMECONVERT_H__87F35906_DBD6_46F1_BF78_2FA34FCB8409__INCLUDED_  
  24.   
  25. #if _MSC_VER > 1000  
  26. #pragma once  
  27. #endif // _MSC_VER > 1000  
  28. #include <time.h>  
  29. #include<olectl.h.>  
  30. class TimeConvert    
  31. {  
  32. public:  
  33.     static void SystemTimeToFileTime(const SYSTEMTIME*pst,FILETIME *pft);  
  34.     static void Time_tToFileName(time_t t,FILETIME *ft);  
  35.     static void TmToTime_t(tm &_t,time_t &t);  
  36.     static void Time_tToTm(time_t t,tm *_t);  
  37.     static void TmToSystemTime(const tm &t,SYSTEMTIME *st);  
  38.     static void SystemTimeTOTm(const SYSTEMTIME &st,tm*t);  
  39.     static void FileTimeToTime_t(const FILETIME  *ft,  time_t  *t);  
  40.     static void Time_tToSystemTime(time_t t, SYSTEMTIME *pst);  
  41.     static void FileTimeToSystemTime(const FILETIME *lpFileTime,SYSTEMTIME *lpSystemTime);  
  42.     static void SystemTimeToTime_T(SYSTEMTIME *st, time_t *pt );  
  43.     static void FileTimeToTm(const FILETIME *ft,tm *t);  
  44.     static void TmToFileTime(tm &t,FILETIME *ft);  
  45.     static void TimeFormat(char *dest,int length,const char *format,const tm*t);  
  46. private:  
  47.     TimeConvert();  
  48.     virtual ~TimeConvert();  
  49.   
  50.   
  51. };  
  52.   
  53. #endif // !defined(AFX_TIMECONVERT_H__87F35906_DBD6_46F1_BF78_2FA34FCB8409__INCLUDED_)  

TimeConvert.cpp
[cpp]  view plain copy
  1. // TimeConvert.cpp: implementation of the TimeConvert class.  
  2. //  
  3. //////////////////////////////////////////////////////////////////////  
  4.   
  5. #include "TimeConvert.h"  
  6.   
  7. //////////////////////////////////////////////////////////////////////  
  8. // Construction/Destruction  
  9. //////////////////////////////////////////////////////////////////////  
  10.   
  11. TimeConvert::TimeConvert()  
  12. {  
  13.   
  14. }  
  15.   
  16. TimeConvert::~TimeConvert()  
  17. {  
  18.   
  19. }  
  20.   
  21. void TimeConvert::FileTimeToSystemTime(const FILETIME *lpFileTime,SYSTEMTIME *lpSystemTime)  
  22. {  
  23.     ::FileTimeToSystemTime(lpFileTime,lpSystemTime);  
  24. }  
  25.   
  26. void TimeConvert::Time_tToSystemTime(time_t t, SYSTEMTIME *pst )  
  27. {  
  28. //  FILETIME ft;  
  29. //  Time_tToFileName(t,&ft);      
  30. //  FileTimeToSystemTime( &ft, pst );  
  31.   
  32.     //另一种实现  
  33.     tm *m=localtime(&t);  
  34.     TmToSystemTime(*m,pst);  
  35. }  
  36. void TimeConvert::SystemTimeToTime_T(SYSTEMTIME *st, time_t *pt )  
  37. {    
  38.     FILETIME ft;  
  39.     SystemTimeToFileTime( st, &ft);   
  40.     FileTimeToTime_t(&ft,pt);  
  41. }  
  42.   
  43. void TimeConvert::FileTimeToTime_t(const FILETIME  *ft,  time_t  *t)  
  44. {  
  45.     LONGLONG  ll;     
  46.     ULARGE_INTEGER ui;    
  47.     ui.LowPart=ft->dwLowDateTime;    
  48.     ui.HighPart=ft->dwHighDateTime;        
  49.     ll=(ft->dwHighDateTime<<32)+ft->dwLowDateTime;        
  50.     *t=(DWORD)((LONGLONG)(ui.QuadPart- 116444736000000000) /10000000);  
  51. }  
  52.   
  53. void TimeConvert::SystemTimeTOTm(const SYSTEMTIME &st,tm*t)  
  54. {  
  55.     tm gm= {st.wSecond, st.wMinute, st.wHour, st.wDay, st.wMonth-1, st.wYear-1900, st.wDayOfWeek, 0, 0};  
  56.     memcpy(t,&gm,sizeof(tm));  
  57. }  
  58. void  TimeConvert::TmToSystemTime(const tm &t,SYSTEMTIME *st)  
  59. {  
  60.     SYSTEMTIME s={1900 + t.tm_year,1 + t.tm_mon,t.tm_wday,t.tm_mday,t.tm_hour,t.tm_min,t.tm_sec,0};  
  61.     memcpy(st,&s,sizeof(SYSTEMTIME));  
  62. }  
  63.   
  64. void TimeConvert::Time_tToTm(time_t t,tm *_t)  
  65. {  
  66.     tm *tt=localtime(&t);  
  67.     memcpy(_t,tt,sizeof(tm));  
  68. }  
  69.   
  70. void TimeConvert::TmToTime_t(tm &_t,time_t &t)  
  71. {  
  72.     t=mktime(&_t);  
  73. }  
  74. void TimeConvert::Time_tToFileName(time_t t,FILETIME *ft)  
  75. {  
  76.     LONGLONG ll = Int32x32To64(t, 10000000) + 116444736000000000;  
  77.     ft->dwLowDateTime = (DWORD) ll;  
  78.     ft->dwHighDateTime = (DWORD)(ll >> 32);      
  79. }  
  80. void TimeConvert::FileTimeToTm(const FILETIME *ft,tm *t)  
  81. {  
  82.     time_t tm;  
  83.     FileTimeToTime_t(ft,&tm);  
  84.     Time_tToTm(tm,t);  
  85. }  
  86. void TimeConvert::TmToFileTime(tm &t,FILETIME *ft)  
  87. {  
  88.     time_t tm;  
  89.     TmToTime_t(t,tm);  
  90.     Time_tToFileName(tm,ft);  
  91. }  
  92. void TimeConvert::SystemTimeToFileTime(const SYSTEMTIME*pst,FILETIME *pft )  
  93. {  
  94.     ::SystemTimeToFileTime(pst,pft);  
  95. }  
  96. void TimeConvert::TimeFormat(char *dest,int length,const char *format,const tm*t)//@param:length 为dest长度-1  
  97. {  
  98.     strftime(dest,length,format,t);  
  99. }  
  100. /* 
  101.  
  102. 这关系太TMD乱了吧.....晕了 
  103.  
  104. 后记总结: 
  105. systemtime与filetime可以通过函数直接转化:FileTimeToSystemTime 和SystemTimeToFileName; 
  106. tm和time_t也可以直接转化:localtime()和mktime(); 
  107. 其他转化需要通过中间步骤 
  108. Sytemtime和tm有些类似,可以通过加减转化(PS:画个图之后,更好理解) 
  109. */  



#include <iostream> #include <string> #include <windows.h> #include <vector> #include <iomanip> #include <sstream> #include <ctime> #include <io.h> using namespace std; // 转换FILETIME到本地时间字符串 string FileTimeToString(const FILETIME& ft) { SYSTEMTIME st; FileTimeToSystemTime(&ft, &st); char buffer[26]; sprintf_s(buffer, "%04d-%02d-%02d %02d:%02d:%02d", st.wYear, st.wMonth, st.wDay, st.wHour, st.wMinute, st.wSecond); return string(buffer); } // 获取文件创建时间 FILETIME GetFileCreationTime(const string& filePath) { HANDLE hFile = CreateFileA(filePath.c_str(), GENERIC_READ, FILE_SHARE_READ, NULL, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, NULL); FILETIME ftCreation = {0}; if (hFile != INVALID_HANDLE_VALUE) { FILETIME ftAccess, ftWrite; if (GetFileTime(hFile, &ftCreation, &ftAccess, &ftWrite)) { // 转换为本地时间(可选) // SYSTEMTIME stLocal; // FileTimeToSystemTime(&ftCreation, &stLocal); } CloseHandle(hFile); } return ftCreation; } // 比较两个FILETIMEtime1 >= time2 返回true) bool CompareFileTime(const FILETIME& time1, const FILETIME& time2) { if (time1.dwHighDateTime > time2.dwHighDateTime) return true; if (time1.dwHighDateTime < time2.dwHighDateTime) return false; return time1.dwLowDateTime >= time2.dwLowDateTime; } // 搜索符合条件的文件 vector<string> SearchFiles(const string& directory, const string& fileType, const FILETIME& startTime, const FILETIME& endTime) { vector<string> result; string searchPath = directory + "\\*.*"; WIN32_FIND_DATAA findData; HANDLE hFind = FindFirstFileA(searchPath.c_str(), &findData); if (hFind != INVALID_HANDLE_VALUE) { do { // 跳过当前目录和上级目录 if (strcmp(findData.cFileName, ".") == 0 || strcmp(findData.cFileName, "..") == 0) { continue; } // 跳过目录 if (findData.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY) { continue; } // 检查文件类型 string fileName = findData.cFileName; size_t dotPos = fileName.find_last_of('.'); string ext = (dotPos != string::npos) ? fileName.substr(dotPos + 1) : ""; if (!fileType.empty() && _stricmp(ext.c_str(), fileType.c_str()) != 0) { continue; } // 检查创建时间 FILETIME creationTime = GetFileCreationTime(directory + "\\" + fileName); if ((CompareFileTime(creationTime, startTime) || startTime.dwHighDateTime == 0) && (CompareFileTime(endTime, creationTime) || endTime.dwHighDateTime == 0)) { result.push_back(directory + "\\" + fileName); } } while (FindNextFileA(hFind, &findData) != 0); FindClose(hFind); } return result; } // 重命名文件 bool RenameFile(const string& oldPath, const string& newPath) { cout << "尝试重命名: " << oldPath << " -> " << newPath << endl; if (_access(oldPath.c_str(), 0) != 0) { cout << "错误: 源文件不存在 - " << oldPath << endl; return false; } if (_access(newPath.c_str(), 0) == 0) { cout << "错误: 目标文件已存在 - " << newPath << endl; return false; } if (MoveFileA(oldPath.c_str(), newPath.c_str())) { cout << "成功: " << newPath << endl; return true; } else { DWORD error = GetLastError(); cout << "错误: 重命名失败 - 错误码: " << error << endl; return false; } } // 转换用户输入的日期为FILETIME FILETIME StringToFileTime(const string& dateStr) { // 格式示例: "2023-01-01 12:00:00" SYSTEMTIME st = {0}; if (dateStr.length() >= 19) { sscanf_s(dateStr.c_str(), "%4d-%2d-%2d %2d:%2d:%2d", &st.wYear, &st.wMonth, &st.wDay, &st.wHour, &st.wMinute, &st.wSecond); FILETIME ft; SystemTimeToFileTime(&st, &ft); return ft; } // 返回0表示不限制时间 return {0}; } int main() { cout << "===== 文件搜索与重命名工具 =====" << endl; // 设置搜索目录(默认为E盘) string directory = "E:\\"; cout << "搜索目录: " << directory << endl; // 获取文件类型筛选条件 string fileType; cout << "请输入文件类型(如txt、jpg,留空则不限制): "; getline(cin, fileType); // 获取时间范围 string startDate, endDate; cout << "请输入开始日期 (格式: YYYY-MM-DD HH:MM:SS,留空则不限制): "; getline(cin, startDate); cout << "请输入结束日期 (格式: YYYY-MM-DD HH:MM:SS,留空则不限制): "; getline(cin, endDate); FILETIME startTime = StringToFileTime(startDate); FILETIME endTime = StringToFileTime(endDate); // 搜索文件 cout << "\n正在搜索文件..." << endl; vector<string> files = SearchFiles(directory, fileType, startTime, endTime); if (files.empty()) { cout << "未找到符合条件的文件!" << endl; system("pause"); return 0; } // 显示搜索结果 cout << "\n找到 " << files.size() << " 个文件:" << endl; for (size_t i = 0; i < files.size(); i++) { FILETIME ft = GetFileCreationTime(files[i]); cout << "[" << setw(2) << i+1 << "] " << files[i] << " (创建于: " << FileTimeToString(ft) << ")" << endl; } // 选择重命名模式 int renameMode; cout << "\n请选择重命名模式:" << endl; cout << "1. 添加前缀" << endl; cout << "2. 添加后缀" << endl; cout << "3. 替换文本" << endl; cout << "4. 序列编号" << endl; cout << "请输入选项 (1-4): "; cin >> renameMode; cin.ignore(); // 清除输入缓冲区 string prefix, suffix, oldText, newText; int startNum = 1; string format; switch (renameMode) { case 1: cout << "请输入要添加的前缀: "; getline(cin, prefix); break; case 2: cout << "请输入要添加的后缀: "; getline(cin, suffix); break; case 3: // 关键修改:将代码块改为单行语句,避免VS2008的语法解析错误 cout << "请输入要替换的文本: "; getline(cin, oldText); cout << "请输入替换后的文本: "; getline(cin, newText); break; case 4: cout << "请输入起始编号: "; cin >> startNum; cin.ignore(); cout << "请输入格式字符串 (例如: img_%%03d): "; getline(cin, format); break; default: cout << "无效的选项!" << endl; system("pause"); return 1; } // 执行重命名 int successCount = 0; int failCount = 0; cout << "\n==== 开始重命名 ====" << endl; for (size_t i = 0; i < files.size(); i++) { const string& filePath = files[i]; size_t lastSlash = filePath.find_last_of("\\/"); size_t lastDot = filePath.find_last_of("."); string path = (lastSlash != string::npos) ? filePath.substr(0, lastSlash + 1) : ""; string fileName = (lastSlash != string::npos) ? filePath.substr(lastSlash + 1) : filePath; string name = (lastDot != string::npos && lastDot > lastSlash) ? fileName.substr(0, lastDot - lastSlash - 1) : fileName; string ext = (lastDot != string::npos && lastDot > lastSlash) ? fileName.substr(lastDot) : ""; string newName = ""; // 使用if-else结构替代复杂的switch-case,提高兼容性 if (renameMode == 1) { newName = prefix + name + ext; } else if (renameMode == 2) { newName = name + suffix + ext; } else if (renameMode == 3) { // 关键修改:将代码块改为单行语句,避免VS2008的语法解析错误 string newFileName = name; size_t pos = 0; while ((pos = newFileName.find(oldText, pos)) != string::npos) { newFileName.replace(pos, oldText.length(), newText); pos += newText.length(); } newName = newFileName + ext; } else if (renameMode == 4) { char buffer[1024]; string fmt = format; size_t pos = 0; while ((pos = fmt.find("%%", pos)) != string::npos) { fmt.replace(pos, 2, "%"); pos += 1; } sprintf_s(buffer, fmt.c_str(), startNum++); newName = buffer + ext; } string newFilePath = path + newName; if (RenameFile(filePath, newFilePath)) { successCount++; } else { failCount++; } } cout << "\n==== 重命名完成 ====" << endl; cout << "成功: " << successCount << " 个" << endl; cout << "失败: " << failCount << " 个" << endl; system("pause"); return 0; }
06-24
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值