1

我正在用 C 语言为 Linux 编写一个程序,它接收一个目录作为参数,然后对于该目录中的每个文件及其每个子目录,调用一个名为 monfile 的程序。这是代码:

#include <stdio.h>
#include <dirent.h>
#include <sys/stat.h>
#include <unistd.h>
#include <signal.h>
#include <stdlib.h>
#include <fcntl.h>

#define DIR_ARG 1
#define DUR_ARG 2
#define SEC_ARG 3
#define LOG_ARG 4
#define OP1_ARG 5
#define OP2_ARG 6

int main(int argc, char** argv)
{
    // Teste aos argumentos
    if (argc < 5) {
        printf("Erro! Argumentos insuficientes!\n");
        return -1;
    }

    // Declaração de variáveis
    DIR *dir;
    struct dirent *dentry;
    struct stat stat_entry;

    int fork_result;


    // Testa se o directório passado como argumento é válido
    if ((dir = opendir( argv[DIR_ARG])) == NULL)
    {
        perror(argv[DIR_ARG]);
        exit(2);
    }

    chdir(argv[DIR_ARG]);
    // Ciclo de propagação
    while ((dentry = readdir(dir)) != NULL) {

        stat(dentry->d_name, &stat_entry);
        // Se for ficheiro regular
        if (S_ISREG(stat_entry.st_mode)) {
            fork_result = fork();
            if (fork_result == -1) {
                printf("file fork error!\n");
                exit(1);
            }
            if (fork_result == 0) {
                execlp("monfile", "monfile", argv[SEC_ARG], dentry->d_name, filedes, (char *)NULL);
                printf("Erro no exec!\n");
                exit(1);
            }
        }
        // Se for directório vai criar um novo processo e passar dir para esse directório 
        if (S_ISDIR (stat_entry.st_mode)) {
            fork_result = fork();
            if (fork_result == -1) {
                printf ("dir fork error!\n");
                exit(1);
            }
            if (fork_result == 0) {
                chdir(dentry->d_name);
                dir = opendir (dentry->d_name);
            }
        }
    }



    return 0;
}

现在,结果是......大量的 exec 错误消息,然后是一堆 fork 错误消息,即使我只是为带有文件和子目录的目录调用它。这给我带来了 2 个问题:a) cicle 是如何进行这么多迭代的?b) 执行程序出了什么问题,看看 monfile 是如何构建的,并且与 mondir 位于同一文件夹中?

所以,我决定通过添加来找出程序正在查看的目录

printf("%s\n", dentry->d_name);

在 cicle 的开头,它以某种方式扫描每个目录,即使它是这样调用的:mondir Subfolder1 ...(其他参数),是与 mondir 在同一目录中的 Subfolder1。目录问题和 exec 问题我在这里做错了什么?

4

3 回答 3

1

readdir 将返回“。” 和“..”所以你的代码将沿着目录树走,并且永远重复每个目录——fork/exec 轰炸你的机器。

此外,如前所述,find -exec这似乎是一个合理的选择?

于 2011-04-02T21:28:13.370 回答
1

您的代码不能很好地处理子目录。您停止处理当前目录(但在此过程中泄漏了一个 DIR 指针,因为在用新目录覆盖之前您没有closedir()对旧目录执行操作),然后chdir()进入新目录。

您也不要避免目录条目...; 您可能不想处理它们,但readdir()会忠实地将它们作为每个目录中的前两个条目返回。这解释了进程的泛滥。

您应该查找 POSIX 函数nftw()并使用它 - 或者在您的代码工作之前不要尝试处理目录。

于 2011-04-02T21:28:26.813 回答
0

您遇到了一个非常常见的错误:返回的名称readdir()没有预先添加源目录名称,因此您必须自己将其放在那里。

于 2011-04-02T21:18:17.390 回答