我一直在尝试使用 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/