下面两个函数可以用来复制一个现存的文件描述符
#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); }
原文:http://blog.csdn.net/aspnet_lyc/article/details/20707115