3

我应该实现一个具有重定向功能的 shell,但是我在如何使用文件描述符方面遇到了麻烦。在我的 test.c 文件中,我正在尝试将测试从“python --version”传输到一个名为“out”的文件中。

int main(int argc, char *argv[], char *const envp[])
{
/***/TOKENIZER *tester;
    int pid = 0;
    int pipefd[2];
    char buffer = 0;

    pipe(pipefd); 
    pid = fork();
    pipefd[1] = open("out", O_WRONLY | O_TRUNC | O_CREAT, S_IRUSR | S_IRGRP | S_IWGRP | S_IWUSR);

    if(pid == 0){
        close(pipefd[1]);
        dup2(pipefd[0], STDIN_FILENO); 
        close(pipefd[0]);
        while(read(STDIN_FILENO, &buffer, 1) > 0){
            write(STDOUT_FILENO, &buffer, 1);
        }

        close(pipefd[0]);
        _exit(EXIT_SUCCESS);
    }else{
        close(pipefd[0]);
        dup2(pipefd[1], STDOUT_FILENO); 
        close(pipefd[1]);

        write(STDOUT_FILENO, "asdsd\n", 6);

        char* ver[2];
        ver[1] = "--version";
        execvp("python", ver);
        exit(EXIT_SUCCESS);
    }
    free(tester);


    return 0;

编辑:意识到我正在混合重定向和管道的部分。这是我尝试运行的一些重定向代码。

int out = open("output.txt", O_WRONLY | O_TRUNC | O_CREAT, S_IRUSR | S_IRGRP | S_IWGRP | S_IWUSR);

dup2(out, 1);
close(out);

char* ver[2];
ver[1] = "--version";
execvp("python", ver);
exit(EXIT_SUCCESS);

return 0;

我试图将 Python 版本控制到 output.txt 中,但我似乎遗漏了一些东西,因为它不起作用。

编辑 2:看起来它正在执行,但它输出到标准输出而不是 output.txt。

int out = open("output.txt", O_WRONLY | O_TRUNC | O_CREAT, S_IRUSR | S_IRGRP | S_IWGRP | S_IWUSR);
dup2(out, STDOUT_FILENO);

char* ver[3];
ver[0] = "python";
ver[1] = "--version";
ver[2] = NULL;
execvp("python", ver);
perror("exec");

return 0;
4

1 回答 1

1
    int pid = 0;
    int pipefd[2];
    char buffer = 0;

    pipe(pipefd); 
    pid = fork();

到现在为止还挺好。您已经创建了一个管道,它的两个文件描述符存储在pipefd[0]and中pipefd[1],并创建了一个子进程。

    pipefd[1] = open("out", O_WRONLY | O_TRUNC | O_CREAT, S_IRUSR | S_IRGRP | S_IWGRP | S_IWUSR);

不好了!您只是用不同的文件描述符覆盖了一个非常好的文件描述符(管道的写入端),指向名为“out”的常规文件。

您是在尝试进行常规文件重定向还是管道?在你尝试实现一个 shell 之前,你至少应该知道这是两个不同的东西。如果您尝试模仿,python --version > out那么您应该注意到该命令行中没有管道字符,那么您为什么会想象管道涉及实现呢?

此外,由于您在 fork 之后执行此操作,因此两个进程现在都在运行,并且它们都将执行打开操作,包括创建和截断副作用。那不可能是你想要的。

我将停在那里,因为在您修复这部分之前,程序的其余部分将没有意义。

但是一些随机的注释:

  • 3-argmain已经过时了一段时间。使用getenv()environ查找环境变量。
  • 单字节缓冲区效率不高,使用更大的缓冲区也不会更难。
  • 这很奇怪——不一定是致命的问题,但绝对不寻常——在父母而不是孩子身上做 exec。
于 2013-05-29T08:00:40.287 回答