C++遍历文件夹

// 示例代码
#include <stdio.h>
#include <io.h>
 
int main (void)
{
    _finddata_t fileDir;
	char* dir="d:\\temp\\*.*";
    long lfDir;
    
	if((lfDir = _findfirst(dir,&fileDir))==-1l)
        printf("No file is found\n");
    else{
        printf("file list:\n");
        do{
			printf("%s\n",fileDir.name);

        }while( _findnext( lfDir, &fileDir ) == 0 );
    }
    _findclose(lfDir);

    return 0;
}


目标:编写程序遍历文件夹及其子文件夹下所有文件,并输出到标准输出流或者文件流。

1. 先考虑在单层目录下,遍历所有文件。以C:\WINDOWS为例:


用到数据结构_finddata_t,文件信息结构体的指针。

struct _finddata_t
{
    unsigned attrib;     //文件属性
    time_t time_create;  //文件创建时间
    time_t time_access;  //文件上一次访问时间
    time_t time_write;   //文件上一次修改时间
    _fsize_t size;  //文件字节数
    char name[_MAX_FNAME]; //文件名
}; 

文件属性是无符号整数,取值为相应的宏:_A_ARCH(存档),_A_SUBDIR(文件夹),_A_HIDDEN(隐藏),_A_SYSTEM(系统),_A_NORMAL(正常),_A_RDONLY(只读)。容易看出,通过这个结构体,我们可以得到关于该文件的很多信息。结合以下函数,我们可以将文件信息存储到这个结构体中:
//按FileName命名规则匹配当前目录第一个文件
_findfirst(_In_ const char * FileName, _Out_ struct _finddata64i32_t * _FindData); 
 //按FileName命名规则匹配当前目录下一个文件
_findnext(_In_ intptr_t _FindHandle, _Out_ struct _finddata64i32_t * _FindData);
 //关闭_findfirst返回的文件句柄
_findclose(_In_ intptr_t _FindHandle);

_findfirst 函数返回的是匹配到文件的句柄,数据类型为long。遍历过程可以指定文件类型,这通过FileName的赋值来实现,例如要遍历C:\WINDOWS下的所有.exe文件。

示例代码:

bool transfer(string fileName = "C:\\Windows\\*.exe", int exeNum = 0)
{
    _finddata_t fileInfo;
    long handle = _findfirst(fileName.c_str(), &fileInfo);

    if (handle == -1L)
    {
        cerr << "failed to transfer files" << endl;
        return false;
    }

    do 
    {
        exeNum ++;
        cout << fileInfo.name <<endl;
    } while (_findnext(handle, &fileInfo) == 0);
    cout << " .exe files' number:  " << exeNum << endl;

    return true;
}

 2. 遍历文件夹及其子文件夹下所有文件。操作系统中文件夹目录是树状结构,使用深度搜索策略遍历所有文件。用到_A_SUBDIR属性,可运行程序如下:

void dfsFolder(string folderPath, ofstream &fout)
{
    _finddata_t FileInfo;
    string strfind = folderPath + "\\*";
    long Handle = _findfirst(strfind.c_str(), &FileInfo);
    
    if (Handle == -1L)
    {
        cerr << "can not match the folder path" << endl;
        exit(-1);
    }
    do{
        //判断是否有子目录
        if (FileInfo.attrib & _A_SUBDIR)    
        {
            //这个语句很重要
            if( (strcmp(FileInfo.name,".") != 0 ) &&(strcmp(FileInfo.name,"..") != 0))   
            {
                string newPath = folderPath + "\\" + FileInfo.name;
                dfsFolder(newPath, fout);
            }
        }
        else  
        {
            fout << folderPath << "\\" << FileInfo.name  << " ";
        }
    }while (_findnext(Handle, &FileInfo) == 0);

    _findclose(Handle);
    fout.close();
}
在判断有无子目录的if分支中,由于系统在进入一个子目录时,匹配到的头两个文件(夹)是"."(当前目录),".."(上一层目录)。需要忽略掉这两种情况。当需要对遍历到的文件做处理时,在else分支中添加相应的代码就好。




### C++ 遍历文件夹的示例代码 以下是几种常见的方法来实现 C++ 文件夹遍历功能: #### 方法一:使用 `FindFirstFile` 和 `FindNextFile`(Windows 平台) 这是 Windows 提供的标准 API,适合用来查找和遍历文件夹中的文件。 ```cpp #include <windows.h> #include <tchar.h> #include <iostream> void traverseDirectory(const TCHAR* path) { WIN32_FIND_DATA findData; HANDLE hFind; _tcscpy_s(findData.cFileName, MAX_PATH, path); _tcscat_s(findData.cFileName, TEXT("\\*")); hFind = FindFirstFile(findData.cFileName, &findData); if (hFind == INVALID_HANDLE_VALUE) { std::cerr << "Error opening directory." << std::endl; return; } do { if (_tcscmp(findData.cFileName, TEXT(".")) && _tcscmp(findData.cFileName, TEXT(".."))) { if (findData.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY) { // 如果是子目录,则递归遍历 TCHAR subPath[MAX_PATH]; _sntprintf_s(subPath, MAX_PATH, TEXT("%s\\%s"), path, findData.cFileName); traverseDirectory(subPath); } else { // 输出文件路径 std::wcout << path << "\\" << findData.cFileName << std::endl; } } } while (FindNextFile(hFind, &findData)); FindClose(hFind); } int main() { const TCHAR* targetDir = TEXT("C:\\YourTargetDirectory"); traverseDirectory(targetDir); return 0; } ``` 这种方法利用了 Windows 的 `FindFirstFile` 和 `FindNextFile` 函数[^3],能够高效地遍历目标文件夹及其子文件夹中的所有文件。 --- #### 方法二:使用 POSIX 接口(跨平台) 如果需要支持 Linux 或其他类 Unix 系统,可以采用标准库 `<dirent.h>` 进行文件夹遍历。 ```cpp #include <dirent.h> #include <stdio.h> #include <string.h> void processFiles(const char* folderPath) { DIR* dir = opendir(folderPath); if (!dir) { fprintf(stderr, "Cannot open directory: %s\n", folderPath); return; } struct dirent* entry; while ((entry = readdir(dir))) { if (strcmp(entry->d_name, ".") == 0 || strcmp(entry->d_name, "..") == 0) continue; char fullPath[1024]; snprintf(fullPath, sizeof(fullPath), "%s/%s", folderPath, entry->d_name); if (entry->d_type == DT_DIR) { // 如果是子目录,则递归遍历 processFiles(fullPath); } else { // 处理文件逻辑 printf("Found file: %s\n", fullPath); } } closedir(dir); } int main() { const char* dataFolder = "/path/to/your/folder"; processFiles(dataFolder); return 0; } ``` 此代码片段展示了如何通过 POSIX 接口访问文件系统,并递归遍历整个目录结构[^2]。 --- #### 方法三:使用 OpenCV 的 `glob()` 函数 OpenCV 库提供了一个简单的工具函数 `cv::glob()`,可以直接用于检索指定模式的文件列表。 ```cpp #include <opencv2/core.hpp> #include <iostream> void listFilesWithGlob(const std::string& pattern) { std::vector<std::string> files; cv::glob(pattern, files, false); for (const auto& file : files) { std::cout << file << std::endl; } } int main() { std::string searchPattern = "C:/YourTargetDirectory/*.jpg"; // 修改为你想要的路径和扩展名 listFilesWithGlob(searchPattern); return 0; } ``` 该方法特别适用于图像处理场景,因为它允许轻松过滤具有特定后缀的文件[^1]。 --- #### 方法四:借助第三方库 Boost.Filesystem Boost 是一个强大的 C++ 扩展库集合,其中 `filesystem` 模块提供了丰富的接口来进行文件操作。 ```cpp #include <boost/filesystem.hpp> #include <iostream> namespace fs = boost::filesystem; void traverseDirectory(fs::path root) { if (!fs::exists(root) || !fs::is_directory(root)) return; for (auto& entry : fs::recursive_directory_iterator(root)) { if (fs::is_regular_file(entry.path())) { std::cout << entry.path().string() << std::endl; } } } int main() { fs::path targetDir("C:/YourTargetDirectory"); traverseDirectory(targetDir); return 0; } ``` 这种方式不仅简洁明了,而且兼容多种操作系统环境[^3]。 --- ### 总结 以上四种方式分别针对不同需求进行了优化设计。具体选择取决于项目运行的目标平台以及开发者的个人偏好。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值