摘要 : 檔案的屬性包括檔案的産生修改和通路時間以及讀寫隐現等,在 DOS 下檔案的屬性可以用 DOS 指令加以修改,在 WINDOWS 下,可以通過右鍵的屬性改變一組檔案和檔案夾的讀寫屬性,但是卻不能修改檔案的産生修改和通路時間,并且不能修改子目錄下的檔案和檔案夾的屬性。本文基于 Windows 的 API 設計了通用的檔案和檔案夾的屬性(包括檔案的産生修改和通路時間以及讀寫隐現等)的修改方法,并實作了其 Visual C++ 程式編碼。
---- 檔案的屬性的修改具有廣泛的使用價值,修改檔案的讀寫隐現等屬性,可以實作檔案的保護和控制,尤其是修改檔案的産生修改和通路時間,可以避免因防病毒更改系統時間而造成的檔案修改時間的紊亂所形成的不便,例如,對程式員來說,編譯時,系統有可能又重新編譯那些修改時間紊亂的檔案,造成不必要的麻煩。
---- 一、應用的API函數
---- 1、API中的檔案函數組提供了檔案讀寫、屬性設定的衆多API函數,在本編碼中用到的該函數組中的函數有:
HANDLE CreateFile( LPCTSTR, DWORD, DWORD,LPSECURITY_ATTRIBUTES,
DWORD,DWORD,HANDLE );
---- 該函數産生或者打開一個系統對象,并傳回一個用以通路該對象的句柄,這些對象可以是檔案、檔案夾、管道等。
HANDLE FindFirstFile( LPCTSTR, LPWIN32_FIND_DATA );
該函數在指定目錄尋找與指定檔案名比對的檔案和檔案夾,并傳回一個查詢句柄。
BOOL FindNextFile( HANDLE,LPWIN32_FIND_DATA );
該函數繼續一個由查詢句柄指定的查詢過程。
BOOL FindClose( HANDLE );
該函數關閉指定的查詢句柄,結束指定的查詢。
DWORD GetFileAttributes( LPCTSTR );
該函數檢取指定的檔案的屬性資訊。
BOOL SetFileAttributes( LPCTSTR,DWORD );
該函數設定指定的檔案的屬性資訊。
---- 2、API中的時間函數組提供了有關時間轉換的衆多API函數,在本編碼中用到的該函數組中的函數有:
BOOL SystemTimeToFileTime( CONST SYSTEMTIME *,LPFILETIME );
該函數将系統時間轉換成檔案時間,檔案時間是一個64位長度的數,
表示從1601年1月1日起的時間偏移,以千萬分之一秒為機關。
BOOL LocalFileTimeToFileTime(CONST FILETIME *,LPFILETIME );
該函數将本時區的檔案時間轉換成格林威治時間的檔案時間。
BOOL SetFileTime( HANDLE,CONST FILETIME*,CONST FILETIME *,CONST FILETIME * );
該函數設定檔案産生修改和通路的時間。
---- 二、編碼實作
---- 1、首先建立一個基于對話框的工程,在對話框窗體中放置以下控制并通過類管理向導為這些控制添加适當變量:
檔案名編輯框,輸入要改變的檔案的全路徑,檔案名中可以有通配符*和?。變量類型CString,名稱m_FILENAME;
修改包括子目錄檔案的檢查框,變量類型BOOL,名稱m_DIRECTORY;
修改檔案通路時間檢查框,變量類型BOOL,名稱m_ACCESS;
檔案通路時間日編輯框,變量類型BOOL,名稱m_ADAY;
檔案通路時間月編輯框,變量類型BOOL,名稱m_AMONTH;
檔案通路時間年編輯框,變量類型BOOL,名稱m_AYEAR;修改檔案産生時間檢查框,變量類型BOOL,名稱m_CREATE;
檔案産生時間日編輯框,變量類型BOOL,名稱m_CDAY;
檔案産生時間時編輯框,變量類型BOOL,名稱m_CHOUR;
檔案産生時間分編輯框,變量類型BOOL,名稱m_CMINUTE;
檔案産生時間月編輯框,變量類型BOOL,名稱m_CMONTH;
檔案産生時間秒編輯框,變量類型BOOL,名稱m_CSECOND;
檔案産生時間年編輯框,變量類型BOOL,名稱m_CYEAR;
修改檔案修改時間檢查框,變量類型BOOL,名稱m_MODIFY;
檔案修改時間日編輯框,變量類型BOOL,名稱m_MDAY;
檔案修改時間時編輯框,變量類型BOOL,名稱m_MHOUR;
檔案修改時間分編輯框,變量類型BOOL,名稱m_MMINUTE;
檔案修改時間月編輯框,變量類型BOOL,名稱m_MMONTH;
檔案修改時間秒編輯框,變量類型BOOL,名稱m_MSECOND;
檔案修改時間年編輯框,變量類型BOOL,名稱m_MYEAR;
修改檔案屬性檢查框,變量類型BOOL,名稱m_PROPERTY;
檔案隐藏屬性檢查框,變量類型BOOL,名稱m_HIDE;
檔案隻讀屬性檢查框,變量類型BOOL,名稱m_READ;
檔案存檔屬性檢查框,變量類型BOOL,名稱m_SAVE;
檔案系統屬性檢查框,變量類型BOOL,名稱m_SYSTEM;
執行更改的按鈕,點選該按鈕後執行標明的更改操作。
以上成員變量可在對話類的構造函數中用目前系統時間予以初始化。
---- 2、在對話類中加入以下成員:
char filter[16];
int ChangeTheProperty(LPSTR Filename,BYTE bFlag=0) ;
其中成員變量filter用以存放帶通配符的檔案名。
成員函數ChangeTheProperty中實作了帶通配符的檔案的屬性的改變,考慮到要周遊子目錄,
該函數采用了遞歸調用,并請注意參數bFlag,其預設值為0,
1表示調用發生自子目錄。其實作如下:
int CChangePropertyDlg::Change
TheProperty(LPSTR FileName,BYTE bFlag)
{
存放路徑
char pathname[MAX_PATH];
存放全路徑檔案名
char allname[MAX_PATH];
初始化pathname
int num=strlen(FileName);
for(int i=num-1;i>=0;i--){
if(FileName[i]=='//')
break;
}
if(i==-1)
return 0;
strncpy(pathname,FileName,i+1);
pathname[i+1]='/0';
查找并修改所有子目錄下符合條件的檔案
DWORD stat=0;
WIN32_FIND_DATA fileinfo;
if(bFlag){
char DFileName[MAX_PATH];
strcpy(DFileName,pathname);
strcat(DFileName,"*.*");
HANDLE handle=FindFirstFile
(DFileName, &fileinfo);
if(handle==INVALID_HANDLE_VALUE)
return 0;
do{
if(fileinfo.cFileName[0]=='.')
continue;
strcpy(allname,pathname);
strcat(allname,fileinfo.cFileName);
//檢取檔案屬性,可以使用
fileinfo的dwFileAttributes成員
stat=GetFileAttributes(allname);
if(stat==0xffffffff){
MessageBox("警告,擷取檔案資訊時出錯!!",
allname, MB_OK|MB_ICONINFORMATION);
continue;
}
//如果查詢到的是檔案夾
if(stat&FILE_ATTRIBUTE_
DIRECTORY && m_DIRECTORY){
char temp[MAX_PATH];
sprintf(temp,"%s%s//%s",
pathname,
fileinfo.cFileName,filter);
//遞歸改變該子目錄下的檔案
ChangeTheProperty(temp,1);
}
}while(FindNextFile
(handle, &fileinfo));
FindClose(handle);
}
查找并修改符合條件的檔案和檔案夾
HANDLE handle=FindFirstFile
( FileName, &fileinfo );
if(handle==INVALID_HANDLE_VALUE)
return 0;
do{
if(fileinfo.cFileName[0]=='.')
continue;
strcpy(allname,pathname);
strcat(allname,fileinfo.cFileName);
//檢取檔案屬性
stat=GetFileAttributes(allname);
if(stat==0xffffffff){
MessageBox("警告,
擷取檔案資訊時出 錯!!",
allname,MB_OK|MB_
ICONINFORMATION);
continue;
}
//設定檔案屬性
BOOL tt=SetFileAttributes(allname,0);
if(!tt){
MessageBox("警告,
檔案資訊出錯!!",
allname,MB_OK|MB_
ICONINFORMATION);
continue;
}
HANDLE hd=CreateFile(allname,
GENERIC_WRITE, FILE_SHARE_WRITE,
NULL,OPEN_EXISTING,
FILE_FLAG_BACKUP_SEMANTICS,NULL);
if(hd!=INVALID_HANDLE_VALUE){
SYSTEMTIME st[3];
GetLocalTime(&st[0]);
st[1]=st[0];st[2]=st[0];
FILETIME ft[3];
FILETIME lft[3];
BYTE ff[3];
ff[0]=0;ff[1]=0;ff[2]=0;
//改變檔案通路時間
if(m_ACCESS){
ff[0]=1;
st[0].wDay =m_ADAY;st[0].
wMonth=m_AMONTH;
st[0].wYear=m_AYEAR;
BOOL bb=SystemTime
ToFileTime(&st[0],&ft[0]);
bb=LocalFileTime
oFileTime(&ft[0],&lft[0]);
}
//改變檔案産生時間
if(m_CREATE){
ff[1]=1;
st[1].wDay=m_CDAY;st[1].
wHour=m_CHOUR;st[1].
wMinute=m_CMINUTE;
st[1].wMonth=m_
CMONTH;st[1].
wSecond=m_CSECOND;
st[1].wYear=m_CYEAR;
BOOL bb=SystemTime
ToFileTime(&st[1],&ft[1]);
bb=LocalFileTime
ToFileTime(&ft[1],&lft[1]);
}
//改變檔案修改時間
if(m_MODIFY){
ff[2]=1;
st[2].wDay=m_MDAY;st[2].
wHour=m_MHOUR;st[2].
wMinute=m_MMINUTE;
st[2].wMonth=m_MMONTH
;st[2].
wSecond=m_MSECOND;
st[2].wYear=m_MYEAR;
BOOL bb=System
TimeToFileTime(&st[2],&ft[2]);
bb=LocalFileTime
ToFileTime(&ft[2],&lft[2]);
}
//改變檔案時間
BOOL tt=SetFile
Time(hd,ff[1]?&lft[1]:
NULL,ff[0]?&lft[0]:
NULL,ff[2]?&lft[2]:NULL);
if(!tt){
MessageBox("警告,檔案資訊出錯!!",
allname,MB_OK|MB_ICONINFORMATION);
}
CloseHandle(hd);
}
//改變檔案屬性
if(m_PROPERTY){
if(m_HIDE)
stat|=FILE_
ATTRIBUTE_HIDDEN;
else
stat&=~FILE_
ATTRIBUTE_HIDDEN;
if(m_READ)
stat|=FILE_
ATTRIBUTE_READONLY;
else
stat&=~FILE_
ATTRIBUTE_READONLY;
if(m_SAVE)
stat|=FILE_
ATTRIBUTE_ARCHIVE;
else
stat&=~FILE_
ATTRIBUTE_ARCHIVE;
if(m_SYSTEM)
stat|=FILE_
ATTRIBUTE_SYSTEM;
else
stat&=~FILE_
ATTRIBUTE_SYSTEM;
}
//設定檔案屬性
tt=SetFileAttributes(allname,stat);
if(!tt){
MessageBox("警告,
檔案資訊出錯!!",allname,
MB_OK|MB_
ICONINFORMATION);
continue;
}
//查找并修改目前目錄
下所有符合條件的檔案
if(stat&FILE_
ATTRIBUTE_DIRECTORY
&& !bFlag){//m_DIRECTORY
char temp[128];
sprintf(temp,"%s%s//%s",
pathname,fileinfo.cFileName,filter);
//遞歸改變該子目錄下的檔案
ChangeTheProperty(temp,1);
continue;
}
}while(FindNextFile( handle, &fileinfo ));
FindClose(handle);
return 1;
}
---- 3、在執行更改的按鈕的BN_CLICKED消息處理器中加入以下代碼,這段代碼首先解析了輸入的檔案名全路徑,并初始化了filter成員變量:
UpdateData(TRUE);
int num=m_FILENAME.GetLength();
for(int i=num-1;i>=0;i--){
if(m_FILENAME.GetAt(i)=='//')
break;
}
if(i!=-1){
CString ss=m_FILENAME.Right( num-1-i );
strcpy(filter,ss.GetBuffer(15) );
ss.ReleaseBuffer();
ChangeTheProperty
( m_FILENAME.GetBuffer(128)) ;
m_FILENAME.ReleaseBuffer();
}
---- 4、編譯連接配接後,在檔案名編輯框中輸入要改變的檔案的帶通配符的全路徑,設定檔案的建立修改通路時間以及隻讀隐藏等屬性,點選該按鈕後執行標明的更改操作。
---- 三、本編碼在NT4.0平台,用VC++6.0編譯,效果良好,符合預期。