关闭它们后重新打开stdout和stdin文件描述符

时间:2012-01-31 17:29:38

标签: c linux unix

我正在编写一个函数,在给定参数的情况下,它会将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);
}

3 个答案:

答案 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()查找代码示例。