C++遍歷某個文件夾下面的子文件夾及其所有文件
方法一
下面是輸出當前目錄下的所有文件夾以及文件的絕對路徑(當然也可以是相對路徑,由輸入的路徑決定),下面的函數接口可以改裝為單輸出文件或者文件夾的接口,這是一個大方面的總接口。
#include <fstream> #include <iostream> #include <string> #include <sstream> #include <vector> using namespace std; void getAllFiles(string path, vector<string>& files) { //文件句柄 long hFile = 0; //文件信息 struct _finddata_t fileinfo; //很少用的文件信息讀取結構 string p; //string類很有意思的一個賦值函數:assign(),有很多重載版本 if ((hFile = _findfirst(p.assign(path).append("\\*").c_str(), &fileinfo)) != -1) { do { if ((fileinfo.attrib & _A_SUBDIR)) //判斷是否為文件夾 { if (strcmp(fileinfo.name, ".") != 0 && strcmp(fileinfo.name, "..") != 0) { files.push_back(p.assign(path).append("/").append(fileinfo.name));//保存文件夾名字 getAllFiles(p.assign(path).append("/").append(fileinfo.name), files);//遞歸當前文件夾 } } else //文件處理 { files.push_back(p.assign(path).append("/").append(fileinfo.name));//文件名 } } while (_findnext(hFile, &fileinfo) == 0); //尋找下一個,成功返回0,否則-1 _findclose(hFile); } } //測試 void main() { string DATA_DIR = "D:/CoderMaker/data_sets/lfw"; vector<string> files; //測試 char * DistAll = "AllFiles.txt"; getAllFiles(DATA_DIR, files);//所有文件與文件夾的路徑都輸出 ofstream ofn(DistAll); //輸出文件流 int size = files.size(); int FaiNum = 0; ofn << size << endl; for (int i = 0; i<size; i++) { ofn << files[i] << endl;
方法二
這次我們講用一種Windows/Linux通用的方法遍歷一個目錄下的所有文件。
Windows/Linux的IDE都會提供一個頭文件——<io.h>。看名字,似乎是關於I/O的,但是實際上它還提供瞭類似於WIN32_FIND_DATA**、FindFirstFile()、**FindNextFile()和FindClose()**的查找文件的功能。
_finddata_t結構
_finddata_t結構用來記錄查找到的文件的信息。實際上有_finddata32_t、_finddata32i64_t、_finddata64i32_t、_finddata64_t、_wfinddata32_t、_wfinddata32i64_t、_wfinddata64i32_t、_wfinddata64_t八個結構,但都隻是在32位/64位整數和字符類型上有所區別,但整體上相同。大致定義如下(MSDN):
struct _finddata_t { unsigned attrib; size_t time_create; size_t time_access; size_t time_write; _fsize_t size; char name[_MAX_PATH]; };
對於不同的_finddata_t結構,time_create、time_access和time_write的類型為_time32_t或_time64_t,size的類型為_fsize_t或__int64,name為char[_MAX_PATH]或wchar_t[_MAX_PATH]。
attrib
unsigned類型,文件屬性。
time_create
_time32_t/_time64_t類型,文件創建時間(FAT文件系統為-1)。以UTC格式存儲,如果需要轉換成當地時間,使用localtime_s()。
time_access
_time32_t/_time64_t類型,文件最後一次被訪問的時間(FAT文件系統為-1)。以UTC格式存儲,如果需要轉換成當地時間,使用localtime_s()。
time_write
_time32_t/_time64_t類型,文件最後一次被寫入的時間。以UTC格式存儲,如果需要轉換成當地時間,使用localtime_s()。
size
_fsize_t/__int64類型,文件的長度(以字節為單位)。
name
char[_MAX_PATH]/wchar_t[_MAX_PATH]類型,文件/目錄名,不包含路徑。
對於不支持文件創建時間、文件上一次訪問時間的文件系統,time_create和time_access為-1。
_MAX_PATH在stdlib.h中被定義為260。
一般_finddata_t被定義為_finddata32_t/_finddata64i32_t,_wfinddata_t被定義為_wfinddata32_t/_wfinddata64i32_t。為方便,下文中將_finddata_t和_wfinddata_t統稱為_finddata_t。
文件屬性常量
一個文件/目錄可以有多種屬性,每種屬性可以是下面列出的屬性之一。
_A_ARCH
檔案。文件被BACKUP指令改變或清除時被設置。值:0x20。
_A_HIDDEN
隱藏。使用DIR指令一般看不到,除非使用/AH選項。值:0x02。
_A_NORMAL
普通。文件沒有更多屬性被設置,可以沒有限制地被讀或寫。值:0x00。
_A_RDONLY
隻讀。不能以“寫”為目的打開該文件,並且不能創建同名的文件。值:0x01。
_A_SUBDIR
子目錄。值:0x10。
_A_SYSTEM
系統文件。使用DIR指令一般看不見,除非使用/A或/A:S選項。值:0x04。
要檢查x是否含有某個屬性a,可以用x & a進行檢查。指定多個屬性可以使用按位or運算符,例如_A_SYSTEM | _A_RDONLY | _A_HIDDEN。
通配符(wildcards)
遍歷文件目錄時需要使用通配符。
_findfirst()/_findnext()/_findclose()函數
_findfirst()函數
intptr_t _findfirst( const char * filespec, struct _finddata_t *fileinfo );
handle
intptr_t類型,搜索句柄。
fileinfo
_finddata_t *類型,函數將會填入文件/目錄信息。
返回值
如果成功,返回0,否則返回-1。如果沒有更多能夠找到的文件瞭,也會導致失敗。
_findclose()函數
int _findclose( intptr_t handle );
關閉搜索句柄並釋放相應的資源。
handle
搜索句柄。
返回值
成功返回0,失敗返回-1。
程序代碼
1. 遍歷目錄下的所有文件
#include <iostream> #include <cstring> // for strcat() #include <io.h> using namespace std; void listFiles(const char * dir); int main() { char dir[200]; cout << "Enter a directory (ends with \'\\\'): "; cin.getline(dir, 200); strcat(dir, "*.*"); // 在要遍歷的目錄後加上通配符 listFiles(dir); return 0; } void listFiles(const char * dir) { intptr_t handle; _finddata_t findData; handle = _findfirst(dir, &findData); // 查找目錄中的第一個文件 if (handle == -1) { cout << "Failed to find first file!\n"; return; } do { if (findData.attrib & _A_SUBDIR && strcmp(findData.name, ".") == 0 && strcmp(findData.name, "..") == 0 ) // 是否是子目錄並且不為"."或".." cout << findData.name << "\t<dir>\n"; else cout << findData.name << "\t" << findData.size << endl; } while (_findnext(handle, &findData) == 0); // 查找目錄中的下一個文件 cout << "Done!\n"; _findclose(handle); // 關閉搜索句柄 }
程序遍歷目錄下的所有文件/目錄,如果是文件則輸出文件大小。
註意_findnext()函數成功返回0,因此要加上==0或!=-1進行判斷,不能省略。
此外還有一個值得註意的地方:
if (findData.attrib & _A_SUBDIR && strcmp(findData.name, ".") && strcmp(findData.name, "..") ) ...
使用_findfirst()、_findnext()進行搜索時,可能會得到”.“和”…”兩個文件夾名。這兩個值可以忽略。
2. 遍歷目錄中的所有文件
註意是“目錄中”而不是“目錄下”,這個程序將會遍歷一個目錄裡包含的所有文件。
#include <iostream> #include <cstring> // for strcpy(), strcat() #include <io.h> using namespace std; void listFiles(const char * dir); int main() { char dir[200]; cout << "Enter a directory: "; cin.getline(dir, 200); listFiles(dir); return 0; } void listFiles(const char * dir) { char dirNew[200]; strcpy(dirNew, dir); strcat(dirNew, "\\*.*"); // 在目錄後面加上"\\*.*"進行第一次搜索 intptr_t handle; _finddata_t findData; handle = _findfirst(dirNew, &findData); if (handle == -1) // 檢查是否成功 return; do { if (findData.attrib & _A_SUBDIR) { if (strcmp(findData.name, ".") == 0 || strcmp(findData.name, "..") == 0) continue; cout << findData.name << "\t<dir>\n"; // 在目錄後面加上"\\"和搜索到的目錄名進行下一次搜索 strcpy(dirNew, dir); strcat(dirNew, "\\"); strcat(dirNew, findData.name); listFiles(dirNew); } else cout << findData.name << "\t" << findData.size << " bytes.\n"; } while (_findnext(handle, &findData) == 0); _findclose(handle); // 關閉搜索句柄 }
到此這篇關於C++獲取某個文件夾下面的子文件夾及其所有文件的文章就介紹到這瞭,更多相關C++遍歷文件夾與子文件夾內容請搜索WalkonNet以前的文章或繼續瀏覽下面的相關文章希望大傢以後多多支持WalkonNet!
推薦閱讀:
- C++編程使用findfirst和findnext查找及遍歷文件實現示例
- c++下使用windows api遍歷指定文件夾及其子文件夾中的文件
- C++編程模板匹配超詳細的識別手寫數字實現示例
- C++學習貝葉斯分類器實現手寫數字識別示例解析
- C語言中“不受限制”的字符串函數總結