Linux的标准输入、标准输出、标准错误默认使用的fd是0、1、2。所谓重定向输入输出,就是更换0、1、2所在的设备为你所需要的设备。
在启动一个新进程的时候,会默认将0、1、2默认设置为虚拟终端/dev/pts/*,这时使用C语言的printf等会将值打印到当前命令行,如果需要打印到其他设备,例如文件、其他终端命令行,只需要将0、1、2指定为你需要文件、其他终端等。
在C编程中,一般使用dup、dup2函数来更改,函数工作内容可以自行百度。
重定向到一个文件
假如我想把进程的输入输出重定向于某个文件,可以这样:
// 打开一个文件,得到文件描述符fd
fd = open("./out.txt", O_RDWR | O_CREAT, S_IRUSR | S_IWUSR | S_IRGRP | S_IWGRP | S_IROTH | S_IWOTH);
if (fd < 0) {
printf("create file failed!\n");
return -1;
} else {
printf("out.txt fd = %d\n", fd);
}
// 备份原来0、1、2所使用的设备,用于还原,如果不需要还原则无所谓
bk_stdin = dup(0);
bk_stdout = dup(1);
bk_stderr = dup(2);
// 重定向
close(0); //释放描述符0所使用的设备,将0空闲出来
//dup(fd); // 不考虑失败可以使用这个,因为dup会从最小能用的值开始复制,有可能当前最小的不是0,导致fd没有复制给0,正常情况下是将fd值复制到0,这样0就指向了这个文件
if(dup2(fd, 0) != 0) { // 手动将fd、0都指向与这个文件
// 错误处理
}
close(1);
//dup(fd);
if(dup2(fd, 1) != 1) {
// 错误处理
}
close(2);
//dup(fd);
if(dup2(fd, 2) != 2) {
// 错误处理
}
以上一波操作后,此时在fprintf(stdout, “”)、fprintf(stdout, “”)、fprintf(stdin, “”)就会默认打印到了文件。
重定向还原
现在想把重定向到文件还原为原来的命令行
// 同理,关闭0,使用dup2函数将原来备份的文件描述符执行于0
close(0);
if (dup2(bk_stdin, 0) != 0) {
printf("revocery stdin failed!\n");
}
close(1);
if (dup2(bk_stdout, 1) != 1) {
printf("revocery stdout failed!\n");
}
close(2);
if (dup2(bk_stderr, 2) != 2) {
printf("revocery stderr failed!\n");
}
// 文件描述符资源回收
close(fd);
close(bk_stdin);
close(bk_stdout);
close(bk_stdout);