gpt4 book ai didi

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

转载 作者:行者123 更新时间:2023-11-30 17:02:36 24 4
gpt4 key购买 nike

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

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com