在Qt中递归遍历目录及其子目录中的所有文件

52

我想要递归地扫描一个目录及其所有子目录,寻找指定扩展名的文件,例如所有的*.jpg文件。在Qt中如何实现?


2
此外,Qt还有一个QFileSystemModel对象,您可能希望考虑研究一下。 - Chris
4个回答

105

我建议你看一下QDirIterator

QDirIterator it(dir, QStringList() << "*.jpg", QDir::Files, QDirIterator::Subdirectories);
while (it.hasNext())
    qDebug() << it.next();

你可以简单地使用QDir :: entryList()进行递归,但是QDirIterator更简单。此外,如果您恰好有具有大量文件的目录,则会从QDir :: entryList()获得相当大的列表,这在小型嵌入式设备上可能不好。

示例(dir为QDir :: currentPath()):

luca @ ~/it_test - [] $ tree
.
├── dir1
│   ├── image2.jpg
│   └── image3.jpg
├── dir2
│   └── image4.png
├── dir3
│   └── image5.jpg
└── image1.jpg

3 directories, 5 files
luca @ ~/it_test - [] $ /path/to/app
"/home/luca/it_test/image1.jpg"
"/home/luca/it_test/dir3/image5.jpg"
"/home/luca/it_test/dir1/image2.jpg"
"/home/luca/it_test/dir1/image3.jpg"

12
由于没有仔细阅读官方文档,我浪费了很多时间。请注意,“构造后,迭代器位于第一个目录条目之前。next()函数返回下一个目录条目的路径并将迭代器前进。”因此,处理文件的代码应该放在it.next()之后。 - Andrew-Dufresne
我经常看到这个代码块,但它不起作用。它只会在目录树的顶层找到与 glob 模式匹配的所有文件,并且不会向下遍历更低级别的目录。 - SixDegrees
@SixDegrees 不太清楚你的意思。我添加了一个示例。 - Luca Carlon

9
这应该可以运行:
void scanDir(QDir dir)
{
    dir.setNameFilters(QStringList("*.nut"));
    dir.setFilter(QDir::Files | QDir::NoDotAndDotDot | QDir::NoSymLinks);

    qDebug() << "Scanning: " << dir.path();

    QStringList fileList = dir.entryList();
    for (int i=0; i<fileList.count(); i++)
    {
        if(fileList[i] != "main.nut" &&
           fileList[i] != "info.nut")
        {
            qDebug() << "Found file: " << fileList[i];
        }
    }

    dir.setFilter(QDir::AllDirs | QDir::NoDotAndDotDot | QDir::NoSymLinks);
    QStringList dirList = dir.entryList();
    for (int i=0; i<dirList.size(); ++i)
    {
        QString newPath = QString("%1/%2").arg(dir.absolutePath()).arg(dirList.at(i));
        scanDir(QDir(newPath));
    }
}

您的代码与以下内容不同:
  • 广度优先搜索而非深度优先搜索(没有理由,我只是更喜欢它)
  • 添加了更多的过滤器以避免符号链接
  • 使用EntryList而非EntryInfoList。如果您只需要文件名,则不需要使用EntryInfoList。

我测试了它并且可以正常工作,但请注意以下几点:

  • 这可能需要很长时间,因此考虑从线程中运行它
  • 如果存在深度递归,则可能会遇到栈问题

除非我漏掉了什么,否则你更改了他的函数输出。他想要给定目录下所有文件的列表,而你的函数只会打印文件名而不提供列表。 - laurent
他没有填充nutFiles字符串列表,所以我没有包含这部分。不过,修改函数并获取所有文件的列表非常容易。 - pnezis

4

我使用了QDirIterator。

以下是我如何做到快速递归查找所有XML绝对文件路径的方法(Qt4.8.1):

// used to store the file paths
filesStack = new QStack<QString>();

// I use a file dialog to let the user choose the root folder to search in
if (fileDialog->exec() == QFileDialog::Accepted) {
    QDir selectedDir(fileDialog->selectedFiles().first());
    selectedDir.setFilter(QDir::Files |
                          QDir::Dirs | QDir::NoDot | QDir::NoDotDot);
    QStringList qsl; qsl.append("*.xml"); // I only want XML files
    selectedDir.setNameFilters(qsl);
    findFilesRecursively(selectedDir);
}

// this function stores the absolute paths of each file in a QVector
void findFilesRecursively(QDir rootDir) {
    QDirIterator it(rootDir, QDirIterator::Subdirectories);
    while(it.hasNext()) {
        filesStack->push(it.next());
    }
}

感谢大家的提示。
编辑:我可能省略了一些声明,请注意。

0

另一个示例,使用QFileInfo索引所有文件:

void ID3Tab::scanDir( QDir dir )
{ QFileInfoList fil = dir.entryInfoList( QStringList( "*.mp3" ),
                                         QDir::Files | QDir::NoDotAndDotDot | QDir::NoSymLinks,
                                         QDir::Name | QDir::IgnoreCase );
  foreach ( QFileInfo fi, fil )
    indexFile( fi );

  QFileInfoList dil = dir.entryInfoList( QStringList( "*" ),
                                         QDir::AllDirs | QDir::NoDotAndDotDot | QDir::NoSymLinks,
                                         QDir::Name | QDir::IgnoreCase );
  foreach ( QFileInfo di, dil )
    scanDir( QDir( di.absoluteFilePath() ) );
}

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接