用windows的API實現文件夾的遍歷掃描的程序
使用到的API:
FindFirstFile
FindNextFile
FindClose
主要用到的結構:
WIN32_FIND_DATA
在msdn都有詳細的說明。這里也簡要說明一下:
HANDLE WINAPI FindFirstFile(
__in LPCTSTR lpFileName, //要查找的文件路徑
__out LPWIN32_FIND_DATA lpFindFileData//保存文件結構的指針
);
查找對應的路徑,如果查找返回查找的句柄,然后我們可以通過該句柄去查找下一個和關閉這個句柄。
如果查找失敗,則返回(INVALED_HANDLE_VALUE)(-1),可以通過GetLastError();來得到他的錯誤類型。(貌似我都沒有這么做過。不過對于嚴謹的程序員們,可能很重要,不過這個也是寫給自己看的,就不考慮那么多了。哈哈)
注意:查找第一個的時候,他不僅返回了查找的句柄,他也把第一個的文件信息保存在lpFindFileData里面了。所以這個也是要處理的。
WIN32_FIND_DATA的結構就是一個結構提,在這里我智慧用到cFileName(文件名)和dwFileAttributes(文件屬性)兩個域。
BOOL WINAPI FindNextFile(
__in HANDLE hFindFile, //查找的文件句柄
__out LPWIN32_FIND_DATA lpFindFileData//保存文件結構的指針
);
他就是在文件句柄的情況下查找下一個文件。并將文件信息保存在lpFindFileData中。成功返回true,失敗返回false(就是沒有下一個了)
BOOL WINAPI FindClose(
__in_out HANDLE hFindFile
);
就是關閉文件句柄咯,沒有什么好說的哈。
要試用的函數介紹完畢。主要的思路:
查找他是要給文件路徑的。對了,只給文件夾路徑還不可以,必須還要加上\\*.*等查找的過濾關鍵字。否則他只找到文件夾,呵呵。過濾的關鍵字也可是?*.*?什么什么的組合都可以可以例如:abc?f*.t?t*這樣。
確定好文件路徑之后,調用FindFirstFile函數來查找,記得要檢查他是否成功哦!還有就是如果成功,第一個查找到的信息已經保存在lpFindFileData里面了。
因為第一個信息已經保存在里面了,而下面還要查找很多下一個,這里必須用到循環了。那么用什么循環呢。看FindNextFile()就知道要用while,不過且慢,上面不是有一個還要處理嘛。難道要先處理完了再用while?no,我們用do while;這樣就可以先處理,然后在循環,直到查找完畢。(之所以寫這些是因為我開始是用的WHILE發現總是少一個,后來才知道first的時候已經有一個了,所以才用了do while哎。。)
循環體:出來單個查找的內容,這里有幾個要注意的是文件夾里面有".." 和"."兩個查找到的不用處理,我也不知道他是干什么的,是網上查到的。呵呵。然后就是判斷是否為文件夾(if(findFileData.dwAttributes & FILE_ATTRIBUTE_DIRECTORY));。否則就不是文件夾,是文件。
然后在分開處理他們。。
最后不要忘了要關閉句柄。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); //找到第一個
17
18 if (hFind == INVALID_HANDLE_VALUE) //如果沒有找到相關的文件信息,返回false
19 {
20 delete pChPath;
21 return false;
22 }
23
24 //處理之后查找下一個,直到都找完
25 do
26 {
27 if (fileFindData.cFileName[0] == '.')
28 {
29 continue; //因為文件夾開始有"."和".."兩個目錄,要過濾掉
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); //遞歸調用來查找他的子目錄
38 }
39
40 //如果是文件
41 pFindProc(pChPath);
42
43 }while (::FindNextFile(hFind, &fileFindData));
44
45 FindClose(hFind); //關閉查找句柄
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 }