标签:
下面两个函数都可用来复制一个现存的文件描述符:
#include<unistd.h>
int dup(int filedes);
int dup2(int filedes,int filedes2);
两函数的返回值:若成功则返回新的文件描述符,若出错则返回-1
由dup返回的新文件描述符一定是当前可用文件描述符中的最小值。用dup2则可以用filedes2参数指定新描述符的数值。如果filedes2已经打开,则现将其关闭。如若filedes等于filedes2,则dup2返回filedes2,而不关闭它。
这些函数返回的新文件描述符与参数filedes共享同一文件表项。如图所示,我们假定执行了:
newfd=dup(1);
当此函数开始执行时,假定下一个可用的描述符是3(这是非常可能的,因为0,1和2由shell打开)。因为两个描述符指向同一文件表项,所以它们共享同一文件状态标志(读、写、添加等)以及同一当前文件偏移量。
每个文件描述符都有它自己的一套文件描述符标志。
复制一个描述符的另一种方式是使用fcntl函数,实际上,可调用:
dup(filedes);
等效于
fcntl(filedes,F_DUPFD,0);
而调用
dup2(filedes,filedes2);
等效于
close(filedes2);
fcntl(filedes,F_DUPFD,filedes2);
在后一种情况下,dup2并不完全等效于close加上fcntl。它们之间的区别是:
1)dup2是一个原子操作,而close及fcntl则包括两个函数调用,有可能在close和fcntl之间插入执行信号捕获函数,它可能修改文件描述符。
2)dup2和fcntl有某些不同的errno。
重点解释两个地方:
第3幅图,要执行dup2(fd, 1);
,文件描述符1原本指向tty
,现在要指向新的文件somefile
,就把原来的关闭了,但是tty
这个文件原本有两个引用计数,还有文件描述符save_fd
也指向它,所以只是将引用计数减1,并不真的关闭文件。
第5幅图,要执行dup2(save_fd, 1);
,文件描述符1原本指向somefile
,现在要指向新的文件tty
,就把原来的关闭了,somefile
原本只有一个引用计数,所以这次减到0,是真的关闭了。
标签:
原文地址:http://www.cnblogs.com/wuchanming/p/4276747.html