Рекурсивно повторите все файлы в каталоге и его подкаталогах в Qt

#qt #qt5 #qdir

Вопрос:

Я хочу рекурсивно сканировать каталог и все его подкаталоги на наличие файлов с заданным расширением-например, все файлы *.jpg. Как вы можете сделать это в Qt?

Комментарии:

1. Кроме того, в Qt есть объект QFileSystemModel, который вы, возможно, захотите изучить

Ответ №1:

Я предлагаю вам взглянуть на 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"
 

Комментарии:

1. Я потратил много времени впустую из-за того, что не читал должным образом официальные документы. Пожалуйста, обратите внимание «После построения итератор находится перед первой записью каталога. next() Функция возвращает путь к следующей записи каталога и продвигает итератор.» Поэтому ваш код, который имеет дело с файлами, должен появиться позже it.next() .

2. Я вижу этот блок кода повсюду, но он не работает. Он находит все файлы, соответствующие шаблону глобуса, только на верхнем уровне дерева каталогов и не опускается на более низкие уровни.

3. @SixDegrees не уверен, что вы имеете в виду. Я добавил пример.

Ответ №2:

Это должно сработать :

 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" amp;amp;
           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));
    }
}
 

Отличия от вашего кода заключаются в следующем:

  • Поиск сначала по ширине, а не по глубине (для этого нет причин, я просто предпочитаю это)
  • Дополнительные фильтры для того, чтобы избежать симуляции ссылок
  • Список участников вместо списка участников. Вам это не нужно, если вам нужно только имя файла.

Я протестировал его, и он работает правильно, но обратите внимание на следующее:

  • Это может занять много времени, поэтому подумайте о том, чтобы запустить его из потока
  • Если есть глубокая рекурсия, у вас могут возникнуть проблемы с вашим стеком

Комментарии:

1. Если я чего-то не упускаю, вы изменили вывод его функции. Ему нужен список всех файлов в данном каталоге, в то время как ваша функция будет печатать только имена файлов, но не будет предоставлять список.

2. Он не заполнял список nutFiles строк, поэтому я не включил эту часть. Однако довольно легко изменить функцию и получить список всех файлов

Ответ №3:

Я использовал 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());
    }
}
 

Спасибо всем за подсказки.

ПРАВКА: возможно, я опустил некоторые заявления, будьте осторожны.

Ответ №4:

Еще один пример, который индексирует все файлы, используя 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() ) );
}