1.描述
复制文件描述符。
dup()
dup()系统调用生成文件描述符oldfd的一个副本,从未使用的文件描述符当中选取最小的文件描述符作为新的文件描述符。
当成功返回后,旧的文件描述符和新的文件描述符可以互换使用,这两个描述符指向了同一个打开的文件描述符(见系统调用open()),他们共享同一个文件偏移量和文件状态标志。比如说,当使用lseek操作某一个文件描述符,改变了该文件的偏移量的时候,使用另外一个描述符看到的是相同的偏移量。
这两个文件描述符不会共享文件描述符标志(例如 close-on-exec标志)。如果旧的文件描述符有close-on-exec标志,则新的文件描述符没有。
dup2()
dup2()系统调用完成和dup相同的功能。但是dup2并不将未使用的文件描述符的最小值作为新的文件描述符,它使用参数newfd作为新的文件描述符。如果newfd在之前已经打开了,会先将打开的文件关闭,在完成文件描述符的复制操作。
关闭newfd之前打开的文件和复制newfd新的文件描述符这个操作是原子的。这个操作的原子性至关重要,因为实现close()和dup()两个函数相同的功能涉及到竞争问题。也就是说在关闭了newfd打开的文件之后,将newfd复制为新的文件描述符之前,newfd有可能已经被重新定义成其他文件的文件描述符了。这种情况有可能会发生,比如说主程序被一个信号打断,或者在完成了close操作之后,又有一个并行执行的线程分配了这个文件描述符。
注意一下要点:
- 如果oldfd不是一个有效的文件描述符,此次调用失败,newfd打开的文件也不关闭
- 如果oldfd是一个有效的文件描述符,并且newfd=oldfd,dup2什么也不做,直接返回newfd()
dup3()
dup3实现和dup2相同的功能,但有以下差别:
-
调用这可以设定close-on-exec标志,这个操作可以通过指定flags参数里面的O_CLOEXEC。参考open系统调用里面相同的标志参数,那里面解释了为什么这样做是有效的。
-
如果oldfd等于newfd,dup3调用失败,伴随这一个错误EINVAL。
2.头文件与声明
#include <unistd.h>
int dup(int oldfd);
int dup2(int oldfd, int newfd);
#define _GNU_SOURCE