3.12 函数dup和dup2
下面两个函授都可用来复制一个现有的文件描述符。
#include <unistd.h> int dup(int fd); int dup2(int fd, int fd2); 量函数的返回值:若成功,返回新的文件描述符;若出错,返回-1. |
由dup返回的新文件描述符一定是当前可用文件描述符中的最小数值。对于dup2,可以用fd2参数指定新描述符的值。如果fd2已经打开,则先将其关闭。如若fd等于fd2,则dup2返回fd2,而不关闭它。否则,fd2的FD_CLOEXEC文件描述符标志就被清除,这样fd2在进程调用exec时是打开状态。
这些函数返回的新文件描述符与参数fd共享同一个文件表项,如图3-9所示。
图略。
在此图中,我们假定进程启动时执行了:
newfd = dup(1);
当此函数开始执行时,假定下一个可用的描述符是3(这是非常有可能的,因为0,1和2都由shell打开)。因为两个描述符指向同一文件表项,所以它们共享同一文件状态标志(读、写、追加等)以及同一当前文件偏移量。
每个文件描述符都有它自己的一套文件描述符标志。正如我们将下一节中说明的那样,新描述符的执行时关闭标志总是由dup函数清除。
复制一个描述符的另一种方法是使用fcntl函数,3.14节将对该函数进行说明。实际上,
调用
dup(fd);
等效于
fcntl(fd, F_DUPFD, 0);
而调用
dup2(fd, fd2);
等效于
close(fd2);
fcntl(fd, F_DUPFD, fd2);
在后一种情况下,dup2并不完全等同于close加上fcntl。它们之间的区别具体如下。
(1)dup2是一个原子操作,而close和fcntl包括两个函数调用。有可能在close和fcntl之间调用了信号捕获函数,它有可能修改文件描述符。如果不同的线程改变了文件描述符的话也会出现相同的问题。
(2)dup2和fcntl有一些不同的errno。