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

Reg*_*ser 32 c unix linux

我正在编写一个函数,在给定参数的情况下,它会将stdout重定向到文件或从文件中读取stdin.为此,我关闭与stdout或stdin关联的文件描述符,这样当我打开文件时,它会在我刚刚关闭的描述符下打开.这是有效的,但问题是,一旦完成,我需要将stdout和stdin恢复到它们真正应该是什么.

我能为stdout做的是打开("/ dev/tty",O_WRONLY); 但我不确定为什么会这样,更重要的是我不知道stdin的等价语句.

对于stdout,我有

close(1);
if (creat(filePath, O_RDWR) == -1)
{
    exit(1);
}
Run Code Online (Sandbox Code Playgroud)

和斯坦丁

close(0);
if (open(filePath, O_RDONLY) == -1)
{
    exit(1);
}
Run Code Online (Sandbox Code Playgroud)

Amb*_*jak 39

您应该使用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);
Run Code Online (Sandbox Code Playgroud)

但是,您可能需要小心处理一些细节(来自man dup):

这两个描述符不共享文件描述符标志(close-on-execflag).关闭重复描述符的close-on-exec标志(FD_CLOEXEC;请参阅fcntl(2)).

如果这是一个问题,您可能必须恢复close-on-exec标志,可能使用dup3()而不是dup2()来避免竞争条件.

另外,请注意,如果您的程序是多线程的,则其他线程可能会意外地写入/读取重新映射的stdin/stdout.

  • 如果你是多线程的还有其他问题 - 如果另一个线程在你的`close(0)`(或`close(1)`)和相应的`open(...)`之间打开一个文件,那么它的文件将会成为stdin(或stdout).使用`dup2()`(或`dup3`)来强制文件描述符为0(或1),而不是依赖于"最低未使用"行为. (2认同)

cni*_*tar 17

我想你可以在重定向之前 "保存"描述符:

int save_in, save_out;

save_in = dup(STDIN_FILENO);
save_out = dup(STDOUT_FILENO);
Run Code Online (Sandbox Code Playgroud)

稍后您可以使用dup2它们来恢复它们:

/* Time passes, STDIN_FILENO isn't what it used to be. */
dup2(save_in, STDIN_FILENO);
Run Code Online (Sandbox Code Playgroud)

我没有在该示例中进行任何错误检查 - 您应该这样做.