c - 重定向标准输出,不理解行为

标签 c io dup2

我一直在尝试使用 dup2 来实现简单的 I/O 重定向。我完全按照其他人说有效的代码进行操作,但仍然没有成功。

int out = open("stdoutput", O_RDWR | O_CREAT | O_TRUNC, S_IRUSR | S_IWUSR);
printf("STANDARD OUT\n");
dup2(out, 1);
printf("SEND TO FILE\n");
dup2(1, out);
close(out);
printf("BACK TO STANDARD OUT\n");

这会将 STANDARD OUT 打印到终端并 发送至文件 返回标准输出 到文件。为什么不切换回来?

我还想将 fork/exec 输出标准输出到文件。

int out2 = open("execout", O_RDWR | O_CREAT | O_TRUNC, S_IRUSR | S_IWUSR);
printf("STANDARD OUT\n");
dup2(out2, 1);
if(fork() == 0)
{
    execvp("time", NULL);
    perror(NULL);
}

dup2(out2, 1);
close(out2);
printf("BACK TO STANDARD OUT\n");

这将打印到终端:

标准输出 TERM_PROGRAM=Apple_Terminal:没有这样的文件或目录 0.00 真实 0.00 用户 0.00 系统

并在文件 execout 中打印: 返回标准输出

对于 time 的 exec 调用似乎存在一些额外的问题(没有找到它,但还是打印它?),但我的主要问题是 I/O 重定向无法正常工作。

预先感谢您的帮助。

<小时/>

更新:似乎有一个问题,特别是 exec 未将输出发送到文件。代码:

int output_file = open("gcc_output", O_RDWR | O_CREAT | O_TRUNC, S_IRUSR | S_IWUSR);

if(fork() == 0)
{
    dup2(output_file, 1);
    close(output_file);
    char *args[] = {"gcc", path, NULL};
    printf("THIS GOES TO FILE");
    execvp(args[0], args); // gcc output, errors etc. go to terminal
    perror("exec");
    exit(2);
}

exec 是否会导致它丢失文件描述符或其他内容?

最佳答案

dup2(from, to) 将文件描述符 from 复制到文件描述符 to;如果 to 打开,它会在被覆盖之前默默关闭。

在第一个示例中,您打开文件 stdoutput 进行写入;它被赋予尽可能低的描述符编号(可能是 3);然后您的 dup2(out2, 1) 将此描述符复制到标准输出描述符 1 上;原来的标准输出被默默关闭了。现在有 4 个打开的描述符:

  • 0是原始标准输入
  • 打开 1 以写入文件 stdoutput
  • 2是原始标准输出
  • 3 打开以写入文件 stdoutput

最后,您尝试通过将文件描述符 1 复制到文件描述符 3 上来“恢复”原始标准输出。

相反,您应该通过 dup原始标准输出存储到另一个文件描述符:

int out = open("stdoutput", O_RDWR | O_CREAT | O_TRUNC, S_IRUSR | S_IWUSR);
printf("STANDARD OUT\n");
int saved_stdout = dup(1); // create a copy of the stdout file descriptor
dup2(out, 1);
close(out);                // close the extra descriptor
printf("SEND TO FILE\n");
dup2(saved_stdout, 1);     // copy the saved stdout file descriptor to 1
close(saved_stdout);       // close the copy of stdout
printf("BACK TO STANDARD OUT\n");
<小时/>

您的 execvp 调用也已损坏;应该有一个额外的参数,即命令名称;另外,time 不会打印当前日期,而是一个需要另一个命令作为参数的命令 - 也许您想执行 date 命令?此外,您希望在 fork 之后的子进程中执行dup2!请尝试以下操作:

int out2 = open("execout", O_RDWR | O_CREAT | O_TRUNC, S_IRUSR | S_IWUSR);
printf("STANDARD OUT\n");
if (fork() == 0) {
    dup2(out2, 1);
    close(out2);
    execvp("date", "date", (char *)NULL);
    perror("exec");
    exit(2);
}
close(out2);
printf("STILL POINTING TO STANDARD OUT\n");

关于c - 重定向标准输出,不理解行为,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/36520903/

相关文章:

c++ - 将 FROM stderr 重定向到另一个文件描述符

c - 如何在 Lua 中访问 C 指针

c - 在 C 中使用 scanf 和 floats

批评我从输入流中跳过 k 个字符的函数

assembly - 如何编写汇编语言使 DMA 工作

scala - Scalaz 中的 Task 和 IO 有什么区别?

c++ - 消失的子进程

python - python 中的 dup、dup2、tmpfile 和 stdout

c++ - 如何从Makefile中识别冗余库?有没有什么工具可以分析库的使用情况?

python - 在 Python 中复制流