在linux下,通过open打开以文件后,会返回一个文件描述符,文件描述符会指向一个文件表,文件表中的节点指针会指向节点表:
其中fd0,1,2分别是标准输入,标准输出,错误输出。
dup和dup2两个函数都可以用来复制打开的文件描述符,复制成功后和复制源共享同一个文件表,如图:
关于dup的测试代码:
#include <unistd.h>
#include <fcntl.h>
#include <stdio.h>
int main(int argc, char *argv[])
{
int fd, copyfd;
fd = open("./log.txt", O_RDWR);
//复制fd
copyfd = dup(fd);
//输出copyfd,应当为4
printf("%d\n", copyfd);
//打印出fd和copyfd的偏移量,都为0
printf("%d\n", (int)lseek(fd, 0, SEEK_CUR));
printf("%d\n", (int)lseek(copyfd, 0, SEEK_CUR));
//将fd的偏移量+3
lseek(fd, 3, SEEK_SET);
//打印出fd和copyfd的偏移量,都为3
printf("%d\n", (int)lseek(fd, 0, SEEK_CUR));
printf("%d\n", (int)lseek(copyfd, 0, SEEK_CUR));
return 0;
}
编译执行上例代码可以发现当移动fd的偏移量时,copyfd的偏移量也发生了变化。往文件里写入内容试试,先把log.txt内容清空:
#include <unistd.h>
#include <fcntl.h>
#include <stdio.h>
int main(int argc, char *argv[])
{
int fd, copyfd;
fd = open("./log.txt", O_RDWR);
//复制fd
copyfd = dup(fd);
char buf1[] = "hello ";
char buf2[] = "world!";
//往fd文件写入内容
if (write(fd, buf1, 6) != 6) {
printf("write error!");
}
//打印出fd和copyfd的偏移量,经过上面的写操作,都变成6了
printf("%d\n", (int)lseek(fd, 0, SEEK_CUR));
printf("%d\n", (int)lseek(copyfd, 0, SEEK_CUR));
//往copyfd写入内容
if (write(copyfd, buf2, 6) != 6) {
printf("write error!");
}
return 0;
}
这样就把helloworld写入文件了,在写入文件的时候,我们使用了两个文件描述符。
dup2测试
#include <unistd.h>
#include <fcntl.h>
#include <stdio.h>
int main(int argc, char *argv[])
{
int fd, copyfd;
fd = open("./log.txt", O_RDWR);
//指定文件描述符号为1000
copyfd = dup2(fd, 1000);
//打印fd和copyfd,应当输出3 1000
printf("%d %d\n", fd, copyfd);
return
上面程序就是指定返回的文件描述为1000,再来看下指定的文件描述符是打开的情况,修改上例代码,将文件描述符指定为1:
copyfd = dup2(fd, 1);
这样就把标准输出重定向到fd描述符指向的文件了。