下面两个函数可以用来复制一个现存的文件描述符
#include <unistd.h>
int dup(int oldfd);
int dup2(int oldfd, int newfd);
由dup返回的新文件描述符一定是当前可用文件描述符的最小值,用dup2则可以用newfd参数指定新描述符的值。如果newfd已经打开,则先将其关闭。如若oldfd等于newfd
则dup2返回newfd,而不关闭它
因为住量描述符指向同一文件表项,所以他们共享同一文件状态标志(读、写、添写、)以及同一当前文件偏移量
复制一个文件描述符的另一种方法是使用fcntl函数
调用dup(fd) 等效于 fcntl(fd, F_DUPFD, 0)
调用dup2(pldfd, newfd) 等效于close(oldfd);fcntl(oldfd, F_DUPFD, newfd);
在后一种情况下,dup2并不完全等同于close加上fcntl。它们之间的区别是:
1.dup2是原子操作而close及fcntl则包括两个函数调用。有可能在close和fcntl之间插入执行信号捕获函数,它可能修改文件描述符
2.dup2和fcntl有某些不同的errno,
下面代码展示了dup函数复制的文件描述符与原文件描述符共享同一文件的特征
#include <stdio.h>
#include <fcntl.h>
#include <stdlib.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <string.h>
main()
{
int fd1;
int fd2;
int r;
char buf[20];
fd1 = open("temp.txt", O_RDWR|O_CREAT|O_EXCL,
S_IRUSR);
if (fd1 == -1)
printf("open error\n"), exit(-1);
r = write(fd1, "Hello ", strlen("Hello "));
if(r != strlen("Hello "))
printf("write error\n"), exit(-1);
fd2 = dup(fd1);
if (fd2 == -1)
printf("dup error\n"), exit(-1);
r = write(fd2, "world\n", strlen("world\n"));
if (r != strlen("world\n"))
printf("write error\n"), exit(-1);
r = read(fd1, buf, sizeof(buf)); //读取失败,因为文件偏移量在末尾
if (r <= 0)
printf("read error1\n"); //read error1将被输出
lseek(fd1, 0, SEEK_SET);
r = read(fd1, buf, sizeof(buf));
if (r <= 0)
printf("read error2\n"), exit(-1);
printf("%s", buf);
close(fd2);
close(fd1);
}