Linux 上readdir 遍历文件夹按文件名排序

本文介绍了如何使用readdir和scandir函数遍历文件夹,并实现文件名的排序输出。详细对比了两种方法的不同,并提供了scandir结合排序函数的具体实现。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

遍历文件夹发送数据,数据文件名按时间排序,使用readdir 遍历时,没有按照文件名排序。
2017-10-15-00-00-00
2017-10-15-01-00-00
2017-10-15-02-00-00

最常见是readdir函数:

DIR   *dir;
struct  dirent *ptr;
dir = opendir(path.c_str());
if(NULL == dir)
{
    cout << "opendir is NULL" << endl;
    return;
}
while( (ptr = readdir(dir))!=NULL)
{
    printf("d_ino:%ld  d_off:%ld d_name: %s\n", ptr->d_ino,ptr->d_off,ptr->d_name);   
}
closedir(dir);

读取/etc/rc.d/目录,输出为:
d_ino:1048669 d_off:856980470154123585 d_name: rc5.d
d_ino:1048663 d_off:884029283848728392 d_name: init.d
d_ino:1048670 d_off:999482920564082138 d_name: rc6.d
d_ino:1059943 d_off:1785792148764494467 d_name: rc
d_ino:1048667 d_off:1841410420704204566 d_name: rc3.d
d_ino:1048666 d_off:3298607144791816940 d_name: rc2.d
d_ino:1048664 d_off:4425734956780595784 d_name: rc0.d
d_ino:1059945 d_off:5016870134774381628 d_name: rc.sysinit
d_ino:1048668 d_off:5182623748018784402 d_name: rc4.d
d_ino:1048665 d_off:6942952852705976198 d_name: rc1.d
d_ino:1048658 d_off:8782171503806864933 d_name: .
d_ino:1059944 d_off:9057741989092678028 d_name: rc.local
d_ino:1048577 d_off:9223372036854775807 d_name: ..

未按照文件名排序。其中:

struct dirent *readdir(DIR *dirp);
struct dirent   
{   
  long d_ino; /* inode number 索引节点号 */  
     
    off_t d_off; /* offset to this dirent 在目录文件中的偏移 */  
    unsigned short d_reclen; /* length of this d_name 文件名长 */  
    unsigned char d_type; /* the type of d_name 文件类型 */  
    char d_name [NAME_MAX+1]; /* file name (null-terminated) 文件名,最长255字符 */  
}

readdir函数按照在磁盘的索引顺序,d_off来排序,如果需要按照文件名d_name,需要遍历后将文件名保存,再次排序。

查阅得知:scandir 可以实现按照规则排序

#include <dirent.h>
int scandir(const char *dirp, struct dirent ***namelist,
    int (*filter)(const struct dirent *),
    int (*compar)(const struct dirent **, const struct dirent **));
int alphasort(const void *a, const void *b);
int versionsort(const void *a, const void *b);

alphasort() and scandir() are specified in POSIX.1-2008, and are widely available. version-sort() is a GNU extension.
The functions scandir() and alphasort() are from 4.3BSD, and have been available under Linux since libc4.

使用scandir遍历文件夹:

struct dirent **namelist;
int n;
n = scandir(path.c_str(),&namelist,0,alphasort);
if(n < 0)
{ 
    cout << "scandir return "<< n  << endl;
}
else
{
    int index=0;
    while(index < n)
    {
        printf("d_ino:%ld  d_off:%ld d_name: %s\n", namelist[index]->d_ino,namelist[index]->d_off,namelist[index]->d_name);
        free(namelist[index]);
        index++;
    }
    free(namelist);
}

输出:
d_ino:1048658 d_off:8782171503806864933 d_name: .
d_ino:1048577 d_off:9223372036854775807 d_name: ..
d_ino:1048663 d_off:884029283848728392 d_name: init.d
d_ino:1059943 d_off:1785792148764494467 d_name: rc
d_ino:1059944 d_off:9057741989092678028 d_name: rc.local
d_ino:1059945 d_off:5016870134774381628 d_name: rc.sysinit
d_ino:1048664 d_off:4425734956780595784 d_name: rc0.d
d_ino:1048665 d_off:6942952852705976198 d_name: rc1.d
d_ino:1048666 d_off:3298607144791816940 d_name: rc2.d
d_ino:1048667 d_off:1841410420704204566 d_name: rc3.d
d_ino:1048668 d_off:5182623748018784402 d_name: rc4.d
d_ino:1048669 d_off:856980470154123585 d_name: rc5.d
d_ino:1048670 d_off:999482920564082138 d_name: rc6.d

scandir函数中可以直接调用排序函数,将遍历到的文件名按照顺序保存在队列中。记得在使用后释放资源。

scandir函数通过参数4 调用比较函数实现排序,可以自定义int (*compar)(const struct dirent , const struct dirent )) 已按照某种规则来实现排序。

git@code.youkuaiyun.com:snippets/2599886.git
https://code.youkuaiyun.com/snippets/2599886

<think>我们面对的问题:使用C++读取指定文件夹下的文件名列表,要求同时支持Windows和Linux系统。 由于Windows和Linux的文件系统接口不同,我们需要使用条件编译(#ifdef)来区分不同平台的处理方式。 参考已有的引用: [^1]提到Linux下获取可执行文件名的方法,其中提到了读取`/proc/self/exe`符号链接,但这里我们关注的是读取目录下的文件名。 [^2]提供了Linux读取目录的方法,使用`dirent`结构体和相关函数(opendir, readdir, closedir)。 对于Windows,我们可以使用Win32 API的`FindFirstFile`和`FindNextFile`函数。 因此,我们可以这样设计: 1. 在Windows平台,使用`FindFirstFile`和`FindNextFile`遍历目录。 2. 在Linux平台,使用`opendir`、`readdir`和`closedir`。 注意:我们需要忽略当前目录(`.`)和上级目录(`..`)。 下面是一个跨平台的实现示例:</think>下面我将详细介绍如何使用 C++ 跨平台读取目录中的文件名列表(支持 Windows 和 Linux 系统): ### 跨平台读取目录文件名的实现方法 #### 1. **使用标准库(C++17 及以上)** C++17 引入了 `<filesystem>` 库,提供跨平台的文件系统操作: ```cpp #include <iostream> #include <filesystem> #include <vector> namespace fs = std::filesystem; std::vector<std::string> getFilenames(const std::string& path) { std::vector<std::string> filenames; for (const auto& entry : fs::directory_iterator(path)) { if (entry.is_regular_file()) { filenames.push_back(entry.path().filename().string()); } } return filenames; } ``` **优点**:代码简洁,跨平台,标准库支持 **缺点**:需要 C++17 或更高版本 #### 2. **平台特定实现(兼容旧标准)** 若需支持 C++11 或更早版本,可使用条件编译: ```cpp #include <iostream> #include <vector> #include <string> #ifdef _WIN32 #include <windows.h> #else #include <dirent.h> #endif std::vector<std::string> getFilenames(const std::string& path) { std::vector<std::string> filenames; #ifdef _WIN32 // Windows 实现 WIN32_FIND_DATAA findData; HANDLE hFind = FindFirstFileA((path + "\\*").c_str(), &findData); if (hFind != INVALID_HANDLE_VALUE) { do { if (!(findData.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY)) { filenames.push_back(findData.cFileName); } } while (FindNextFileA(hFind, &findData)); FindClose(hFind); } #else // Linux/Unix 实现 (参考引用[2]) DIR *dir = opendir(path.c_str()); if (dir) { struct dirent *ent; while ((ent = readdir(dir)) != nullptr) { if (ent->d_type == DT_REG) { // 只添加普通文件 filenames.push_back(ent->d_name); } } closedir(dir); } #endif return filenames; } ``` #### 3. **使用示例** ```cpp int main() { std::string directory = "/path/to/directory"; // 替换为实际路径 auto files = getFilenames(directory); std::cout << "找到 " << files.size() << " 个文件:\n"; for (const auto& file : files) { std::cout << "- " << file << "\n"; } return 0; } ``` ### 关键注意事项 1. **路径格式**: - Windows 使用反斜杠 `\`(如 `C:\\Folder`) - Linux 使用正斜杠 `/`(如 `/home/user`) - 建议使用 `std::filesystem::path` 自动处理路径差异 2. **文件类型过滤**: - 使用 `d_type == DT_REG`(Linux)或 `FILE_ATTRIBUTE_DIRECTORY`(Windows)排除目录 - 如需包含子目录,需实现递归遍历 3. **错误处理**: - 检查目录是否存在:`fs::exists(path)`(C++17) - 处理权限问题:捕获异常或检查 API 返回值 4. **排序输出**: ```cpp std::sort(files.begin(), files.end()); // 按文件名排序 ``` ### 性能比较 | 方法 | 跨平台性 | 标准支持 | 实现复杂度 | |-------------------|----------|----------|------------| | C++17 <filesystem>| ★★★★★ | C++17+ | ★☆☆☆☆ | | 平台 API | ★★★☆☆ | 无要求 | ★★★★☆ | > **建议**:优先使用 C++17 的 `<filesystem>`,如需兼容旧系统可使用平台 API 实现[^1][^2]。
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值