用windows的API實(shí)現(xiàn)文件夾的遍歷掃描的程序
使用到的API:
FindFirstFile
FindNextFile
FindClose
主要用到的結(jié)構(gòu):
WIN32_FIND_DATA
在msdn都有詳細(xì)的說明。這里也簡要說明一下:
HANDLE WINAPI FindFirstFile(
__in LPCTSTR lpFileName, //要查找的文件路徑
__out LPWIN32_FIND_DATA lpFindFileData//保存文件結(jié)構(gòu)的指針
);
查找對(duì)應(yīng)的路徑,如果查找返回查找的句柄,然后我們可以通過該句柄去查找下一個(gè)和關(guān)閉這個(gè)句柄。
如果查找失敗,則返回(INVALED_HANDLE_VALUE)(-1),可以通過GetLastError();來得到他的錯(cuò)誤類型。(貌似我都沒有這么做過。不過對(duì)于嚴(yán)謹(jǐn)?shù)某绦騿T們,可能很重要,不過這個(gè)也是寫給自己看的,就不考慮那么多了。哈哈)
注意:查找第一個(gè)的時(shí)候,他不僅返回了查找的句柄,他也把第一個(gè)的文件信息保存在lpFindFileData里面了。所以這個(gè)也是要處理的。
WIN32_FIND_DATA的結(jié)構(gòu)就是一個(gè)結(jié)構(gòu)提,在這里我智慧用到cFileName(文件名)和dwFileAttributes(文件屬性)兩個(gè)域。
BOOL WINAPI FindNextFile(
__in HANDLE hFindFile, //查找的文件句柄
__out LPWIN32_FIND_DATA lpFindFileData//保存文件結(jié)構(gòu)的指針
);
他就是在文件句柄的情況下查找下一個(gè)文件。并將文件信息保存在lpFindFileData中。成功返回true,失敗返回false(就是沒有下一個(gè)了)
BOOL WINAPI FindClose(
__in_out HANDLE hFindFile
);
就是關(guān)閉文件句柄咯,沒有什么好說的哈。
要試用的函數(shù)介紹完畢。主要的思路:
查找他是要給文件路徑的。對(duì)了,只給文件夾路徑還不可以,必須還要加上\\*.*等查找的過濾關(guān)鍵字。否則他只找到文件夾,呵呵。過濾的關(guān)鍵字也可是?*.*?什么什么的組合都可以可以例如:abc?f*.t?t*這樣。
確定好文件路徑之后,調(diào)用FindFirstFile函數(shù)來查找,記得要檢查他是否成功哦!還有就是如果成功,第一個(gè)查找到的信息已經(jīng)保存在lpFindFileData里面了。
因?yàn)榈谝粋€(gè)信息已經(jīng)保存在里面了,而下面還要查找很多下一個(gè),這里必須用到循環(huán)了。那么用什么循環(huán)呢。看FindNextFile()就知道要用while,不過且慢,上面不是有一個(gè)還要處理嘛。難道要先處理完了再用while?no,我們用do while;這樣就可以先處理,然后在循環(huán),直到查找完畢。(之所以寫這些是因?yàn)槲议_始是用的WHILE發(fā)現(xiàn)總是少一個(gè),后來才知道first的時(shí)候已經(jīng)有一個(gè)了,所以才用了do while哎。。)
循環(huán)體:出來單個(gè)查找的內(nèi)容,這里有幾個(gè)要注意的是文件夾里面有".." 和"."兩個(gè)查找到的不用處理,我也不知道他是干什么的,是網(wǎng)上查到的。呵呵。然后就是判斷是否為文件夾(if(findFileData.dwAttributes & FILE_ATTRIBUTE_DIRECTORY));。否則就不是文件夾,是文件。
然后在分開處理他們。。
最后不要忘了要關(guān)閉句柄。FileClose(hFind);
主要思路就是這些了。呵呵。。源代碼奉上:
1 #include<iostream>
2 #include <Windows.h>
3 #include <string>
4 #include <assert.h>
5
6 typedef bool (*FindProc)(const char* chFinleName);
7
8 //遍歷文件夾,chFolderPath為要遍歷的文件夾路徑。chFilter為遍歷文件夾的過濾器,例如*.txt等
9 bool FindFolder(const char* chFolderPath, const char* chFilter, bool bFindFolder, FindProc pFindProc)
10 {
11 assert(pFindProc);
12 int nPathLen = strlen(chFolderPath) + MAX_PATH;
13 char* pChPath = new char[nPathLen];
14 sprintf_s(pChPath, nPathLen, "%s\\%s", chFolderPath, chFilter);
15 WIN32_FIND_DATA fileFindData;
16 HANDLE hFind = ::FindFirstFile(pChPath, &fileFindData); //找到第一個(gè)
17
18 if (hFind == INVALID_HANDLE_VALUE) //如果沒有找到相關(guān)的文件信息,返回false
19 {
20 delete pChPath;
21 return false;
22 }
23
24 //處理之后查找下一個(gè),直到都找完
25 do
26 {
27 if (fileFindData.cFileName[0] == '.')
28 {
29 continue; //因?yàn)槲募A開始有"."和".."兩個(gè)目錄,要過濾掉
30 }
31
32 sprintf_s(pChPath, nPathLen, "%s\\%s", chFolderPath, fileFindData.cFileName); //文件的完整路徑
33
34 //如果要查找遞歸查找文件夾
35 if (bFindFolder && (fileFindData.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY))
36 {
37 FindFolder(pChPath, chFilter, bFindFolder, pFindProc); //遞歸調(diào)用來查找他的子目錄
38 }
39
40 //如果是文件
41 pFindProc(pChPath);
42
43 }while (::FindNextFile(hFind, &fileFindData));
44
45 FindClose(hFind); //關(guān)閉查找句柄
46 delete pChPath;
47 return true;
48 }
49
50 bool PrintFiles(const char* chFileName)
51 {
52 printf("%s\n", chFileName);
53 return true;
54 }
55 int main()
56 {
57
58 FindFolder("test", "*.*", true, &PrintFiles);
59 getchar();
60
61 return 0;
62 }