思路和方法
我们在Windows环境下开发某些具有数据备份和恢复等功能的软件时,需要在拷贝文件及其目录时把文件和目录的所有属性,包括日期和时间都能完全地保存并还原出来。但我们发现,在Windows环境下没有提供任何方法和手段让我们修改一个目录的日期和时间。在查找Win32
API时也只能看到修改文件时间的API函数调用,没有关于修改目录时间的任何描述。在查看Windows提供的备份程序的功能实现时,发现它可以原样地恢复所有子目录的时间(很遗憾,根目录的时间没有恢复),于是我们就从与备份有关的Win32
API入手,终于找到解决之道:就是以“备份”方式,像打开文件那样打开目录。具体来说,就是以FILE—FLAG—BACKUP—SEMANTICS属性来调用CreateFile()
Win32 API函数来打开目录,然后再调用和修改文件时间一样的SetFileTime()
Win32 API函数来修改目录的时间即可。
这样,在我们的备份和恢复程序中,所有目录(包括根目录)都可以完全恢复原来的日期和时间了。同样的技术也可以用于其它的用途,如可以考虑给Windows的资源管理器增加“修改目录时间”的功能等。
具体实现修改
以下是具体实现的VC++ 6.0源代码,它首先将指定目录(如“C:\dir”)当前的时间显示出来,然后把它修改成指定时间,最后把修改后的目录的时间显示出来。
BOOL SetDirTime(char ?DirName, SYSTEMTIME new—stime) //
修改指定目录的时间
{ HANDLE hDir;
hDir = CreateFile ( // 打开目录的Win32 API调用
DirName,
GENERIC—READ | GENERIC—WRITE,
// 必须“写”方式打开
FILE—SHARE—READ|FILE—SHARE—DELETE,NULL,OPEN—EXISTING,
// 打开现存的目录
FILE—FLAG—BACKUP—SEMANTICS,
// 只有这样才能打开目录
NULL);
if (hDir ==INVALID—HANDLE—VALUE )
return FALSE; // 打开失败时返回
FILETIME lpCreationTime;
// creation time目录的创建时间
FILETIME lpLastAclearcase/" target="_blank" >ccessTime; // last access time最近一次访问目录的时间
FILETIME lpLastWriteTime; // last write time最近一次修改目录的时间
SystemTimeToFileTime(&new—stime, &lpCreationTime); // 转换成文件的时间格式
SystemTimeToFileTime(&new—stime, &lpLastAccessTime);
SystemTimeToFileTime(&new—stime, &lpLastWriteTime);
BOOL retval = SetFileTime( // 修改目录时间的Win32 API函数调用
hDir, &lpCreationTime,&lpLastAccessTime,&lpLastWriteTime);
CloseHandle(hDir); // 关闭目录
return retval;
// 返回修改成功与否的返回码
}
BOOL GetDirTime(char ?DirName, SYSTEMTIME & stime) //
获取指定目录的时间
{ HANDLE hDir;
hDir = CreateFile ( // 打开目录的Win32 API调用
DirName, GENERIC—READ,
// 只需读方式打开即可
FILE—SHARE—READ|FILE—SHARE—DELETE, NULL, OPEN—EXISTING,
// 打开现存的目录
FILE—FLAG—BACKUP—SEMANTICS,NULL);
FILETIME lpCreationTime;
// creation time目录创建时间
FILETIME lpLastAccessTime;
// last access time目录最近访问时间
FILETIME lpLastWriteTime;
// last write time目录最近修改时间
BOOL retval = GetFileTime(
// 获取目录日期和时间的Win32 API调用
hDir,&lpCreationTime,&lpLastAccessTime &lpLastWriteTime);
if ( retval ){
FILETIME ftime;
FileTimeToLocalFileTime(&lpLastWriteTime, &ftime); // 转换成本地时间
FileTimeToSystemTime(&ftime, &stime) ; // 转换成系统时间格式}
CloseHandle(hDir);
return retval;}
int DoTest(char ?DirName)
{ SYSTEMTIME stime;
printf(″testing for directory [%s]\n″, DirName);
if ( GetDirTime(DirName, stime) )
// 显示修改前目录的时间
printf(″before change is %04d-%02d-%02d %02d:%02d:%02d\n″,
stime.wYear , stime.wMonth , stime.wDay ,
stime.wHour , stime.wMinute, stime.wSecond );
else
printf(″failed to get the datetime of directory...\n″);
stime.wYear = 1995;
stime.wMonth = 5;
stime.wDay = 12;
stime.wHour = 10 - 8; // GMT time, GMT+8 for China PRC
stime.wMinute = 11;
stime.wSecond = 12;
// GetSystemTime(&stime);
// 如果要设置成当前的时间
if ( SetDirTime(DirName, stime) )
// 修改目录的时间
printf(″success to change datetime of directory.\n″);
else
printf(″failed to change the datetime of directory...\n″);
if ( GetDirTime(DirName, stime) )
// 显示修改后目录的时间
printf(″after change is %04d-%02d-%02d %02d:%02d:%02d\n″,stime.wYear ,
stime.wMonth , stime.wDay ,stime.wHour , stime.wMinute, stime.wSecond );
else
printf(″failed to get the datetime of directory...\n″);
return 0;}
void main(int argc, char? argv[])
{ DoTest(″c:\\dir″);}