一、dup 和 dup2 函数
这两个函数都可以来复制一个现有的文件描述符,他们的声明如下:
#include <unistd.h>
int dup(int fd);
int dup2(int fd, int fd2);
关于 dup 函数,当我们调用它的时候,dup 会返回一个新的描述符,这个描述一定是当前可用文件描述符中的最小值。
我们知道,一般的 0、1、2 描述符分别被标准输入、输出、错误占用,所以在程序中如果 close 掉标准输出 1 后,调用 dup 函数,此时返回的描述符就是 1 。
对于 dup2,可以用 fd2 指定新描述符的值,如果 fd2 本身已经打开了,则会先将其关闭。如果 fd 等于 fd2,则返回 fd2,并不关闭它。
这两个函数返回的描述符与 fd 描述符所指向的文件共享同一文件表项。如下图所示:
也就是 fd 与 fd2 可对同一个文件进行读写操作。且其是一种原子操作。
二、重定向示例
1、duptest.c
#include <fcntl.h>
#include <stdio.h>
#include <stdlib.h>
#include <sys/stat.h>
#include <unistd.h>
#include <string.h>
int main(int argc, char *argv[])
{
int i_fd = open("hello.txt", O_CREAT | O_APPEND | O_RDWR, 0666);
if (i_fd < 0) {
printf("open error!\n");
return 0;
}
const char *buf = "hello fd\n";
int bufLen = strlen(buf);
if (write(i_fd, buf, bufLen) != bufLen) {
printf("write fd error\n");
}
printf("write fd char len is %d\n", bufLen);
int i_dup_fd = dup(i_fd);
if (i_dup_fd < 0) {
printf("dup error!\n");
return 0;
}
printf("i_dup_fd = %d \t i_fd = %d\n", i_dup_fd, i_fd);
close(i_fd);
const int bufLen1 = 10;
char c_buffer[bufLen1];
int n = 0;
while ((n = read(STDIN_FILENO, c_buffer, bufLen1)) != 0) {
printf("write dup fd char count is %d!\n", n);
if (write(i_dup_fd, c_buffer, n) != n) {
printf("write dup fd error!\n");
return 0;
}
}
return 0;
}
如上打开一个文件,我们先写入文件内容“hello fd”,然后将 fd 的描述符拷贝到 dup_fd 的文件描述符上,然后将标准输入的内容写入到 dup_fd 的文件中。
运行程序如下:
查看文件可以看到文件内容如下:
2、dup2test.c
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <sys/stat.h>
#include <fcntl.h>
int main(int argc, char *argv[])
{
int i_fd = open("hello_dup2.txt", O_CREAT | O_APPEND | O_RDWR, 0666);
if (i_fd < 0) {
printf("open error!\n");
return 0;
}
const char *buf = "hello i_fd\n";
const int bufLen = strlen(buf);
if (write(i_fd, buf, bufLen) != bufLen) {
printf("write dup2 error\n");
}
int i_dup2_fd = dup2(i_fd, STDOUT_FILENO);
if (i_dup2_fd != STDOUT_FILENO) {
printf("error dup2!\n");
return 0;
}
close(i_fd);
char c_buf[1024];
int i_read_n = 0;
while ((i_read_n = read(STDIN_FILENO, c_buf, 1024)) != 0) {
i_read_n = read(STDIN_FILENO, c_buf + i_read_n, sizeof(c_buf) - 1 - i_read_n);
if (i_read_n < 0) {
printf("read error!\n");
return 0;
}
printf("%s", c_buf);
fflush(stdout);
sleep(1);
}
close(i_dup2_fd);
return 0;
}
如上,这里没有像使用 dup 的时候显示的调用 write 函数将标准输入的内容写入到指定文件中,而是将标准输出重定向到指定文件中,然后调用 printf 函数将标准输出的内容重定向到指定文件中。
我们在写简单的日志时就可以将 printf 的内容重定向到日志中,使用 printf 作为写日志的接口。
如上运行程序如下:
查看 hello_dup2.txt 可以看到如下:
(SAW:Game Over!)