рекурсивное сканирование папок на c++
Я хочу сканировать дерево каталогов и перечислить все файлы и папки внутри каждого каталога. Я создал программу, которая загружает изображения из веб-камеры и сохраняет их локально. Эта программа создает filetree на основе времени загрузки изображения. Теперь я хочу сканировать эти папки и загружать изображения на веб-сервер, но я не уверен, как я могу сканировать каталоги, чтобы найти изображения. Если бы кто-нибудь мог опубликовать образец кода, это было бы очень полезный.
редактировать: Im работает на встроенной системе linux и не хочет использовать boost
6 ответов
посмотреть man ftw
для простой "прогулки по дереву файлов". Я также использовал fnmatch
в этом примере.
#include <ftw.h>
#include <fnmatch.h>
static const char *filters[] = {
"*.jpg", "*.jpeg", "*.gif", "*.png"
};
static int callback(const char *fpath, const struct stat *sb, int typeflag) {
/* if it's a file */
if (typeflag == FTW_F) {
int i;
/* for each filter, */
for (i = 0; i < sizeof(filters) / sizeof(filters[0]); i++) {
/* if the filename matches the filter, */
if (fnmatch(filters[i], fpath, FNM_CASEFOLD) == 0) {
/* do something */
printf("found image: %s\n", fpath);
break;
}
}
}
/* tell ftw to continue */
return 0;
}
int main() {
ftw(".", callback, 16);
}
(даже не проверено компиляцией, но вы получаете идею.)
это гораздо проще, чем иметь дело с DIRENT
S и рекурсивный обход себя.
для большего контроля над пересечением, есть также fts
. В этом примере dot-files (файлы и каталоги с именами, начинающимися с ".") АР пропущено, если явно не передано программе в качестве отправной точки.
#include <fts.h>
#include <string.h>
int main(int argc, char **argv) {
char *dot[] = {".", 0};
char **paths = argc > 1 ? argv + 1 : dot;
FTS *tree = fts_open(paths, FTS_NOCHDIR, 0);
if (!tree) {
perror("fts_open");
return 1;
}
FTSENT *node;
while ((node = fts_read(tree))) {
if (node->fts_level > 0 && node->fts_name[0] == '.')
fts_set(tree, node, FTS_SKIP);
else if (node->fts_info & FTS_F) {
printf("got file named %s at depth %d, "
"accessible via %s from the current directory "
"or via %s from the original starting directory\n",
node->fts_name, node->fts_level,
node->fts_accpath, node->fts_path);
/* if fts_open is not given FTS_NOCHDIR,
* fts may change the program's current working directory */
}
}
if (errno) {
perror("fts_read");
return 1;
}
if (fts_close(tree)) {
perror("fts_close");
return 1;
}
return 0;
}
опять же, это не компилируется и не тестируется, но я подумал, что упомяну об этом.
импульс.Файловая система позволяет вам это делать. Проверьте docs!
EDIT:
Если вы используете Linux и не хотите использовать Boost,вам придется использовать собственные функции Linux C. на этой странице показывает много примеров того, как это сделать.
Я старая школа, нет ftw () для меня! Это грубо (прошло некоторое время с тех пор, как я занимался прямым программированием на C), и многие вещи жестко закодированы, и я, вероятно, испортил свои вычисления длины для функций strnc* (), но вы поняли идею. Аналогичный пример есть в K&R btw.
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <sys/types.h>
#include <dirent.h>
void listdir(char* dirname, int lvl);
int main(int argc, char** argv)
{
if (argc != 2) {
fprintf(stderr, "Incorrect usage!\n");
exit(-1);
}
listdir(argv[1], 0);
return 0;
}
void listdir(char* dirname, int lvl)
{
int i;
DIR* d_fh;
struct dirent* entry;
char longest_name[4096];
while( (d_fh = opendir(dirname)) == NULL) {
fprintf(stderr, "Couldn't open directory: %s\n", dirname);
exit(-1);
}
while((entry=readdir(d_fh)) != NULL) {
/* Don't descend up the tree or include the current directory */
if(strncmp(entry->d_name, "..", 2) != 0 &&
strncmp(entry->d_name, ".", 1) != 0) {
/* If it's a directory print it's name and recurse into it */
if (entry->d_type == DT_DIR) {
for(i=0; i < 2*lvl; i++) {
printf(" ");
}
printf("%s (d)\n", entry->d_name);
/* Prepend the current directory and recurse */
strncpy(longest_name, dirname, 4095);
strncat(longest_name, "/", 4095);
strncat(longest_name, entry->d_name, 4095);
listdir(longest_name, lvl+1);
}
else {
/* Print some leading space depending on the directory level */
for(i=0; i < 2*lvl; i++) {
printf(" ");
}
printf("%s\n", entry->d_name);
}
}
}
closedir(d_fh);
return;
}
вы захотите использовать функции каталога, объявленные в dirent.h. Это Википедия страницы описывает их и включает образцы кода. Для вашего приложения, как только вы определили каталог, вы захотите снова вызвать функцию обработки рекурсивно для обработки содержимого каталога.
Я думаю, если вы можете использовать Qt / Embedded, есть классы QDir и QFileInfo, которые может помочь вам, хотя это зависит от того, можете ли вы использовать Qt. Вопрос в том, какой API предоставляет ваша система.