我正在编写一个函数,在给定参数的情况下,它会将stdout重定向到文件或从文件中读取stdin。为此,我关闭与stdout或stdin关联的文件描述符,这样当我打开文件时,它会在我刚刚关闭的描述符下打开。这是有效的,但问题是,一旦完成,我需要将stdout和stdin恢复到它们真正应该的位置。
我能为stdout做些什么是打开的(“/ dev / tty”,O_WRONLY);但我不确定为什么会这样,更重要的是我不知道stdin的等价声明。
所以我有,对于stdout
close(1);
if (creat(filePath, O_RDWR) == -1)
{
exit(1);
}
和stdin
close(0);
if (open(filePath, O_RDONLY) == -1)
{
exit(1);
}
答案 0 :(得分:40)
您应该使用dup()和dup2()来克隆文件描述符。
int stdin_copy = dup(0);
int stdout_copy = dup(1);
close(0);
close(1);
int file1 = open(...);
int file2 = open(...);
< do your work. file1 and file2 must be 0 and 1, because open always returns lowest unused fd >
close(file1);
close(file2);
dup2(stdin_copy, 0);
dup2(stdout_copy, 1);
close(stdin_copy);
close(stdout_copy);
但是,您可能需要小心处理一些细节(来自man dup):
这两个描述符不共享文件描述符标志( 特写上execflag)。 close-on-exec标志(FD_CLOEXEC;参见fcntl(2)) 重复的描述符已关闭。
如果这是一个问题,你可能需要恢复close-on-exec标志,可能使用dup3()而不是dup2()来避免竞争条件。
另外,请注意,如果您的程序是多线程的,其他线程可能会意外地写入/读取您重新映射的stdin / stdout。
答案 1 :(得分:17)
我认为你可以在重定向之前“保存”描述符 :
int save_in, save_out;
save_in = dup(STDIN_FILENO);
save_out = dup(STDOUT_FILENO);
稍后您可以使用dup2
恢复它们:
/* Time passes, STDIN_FILENO isn't what it used to be. */
dup2(save_in, STDIN_FILENO);
我没有在该示例中进行任何错误检查 - 您应该。
答案 2 :(得分:1)
您可以创建子进程,并仅在子进程内设置重定向。然后等待子进程终止,并继续在父进程中工作。这样你根本不必担心扭转你的重定向。
只需使用fork()和wait()查找代码示例。