1、dup函数和dup2函数
#include <unistd.h>
int dup(int filedes);
int dup2(int filedes1, int filedes2);
//两个函数,若成功则返回新的文件描述符,若出错则返回-1,同时设置errno
对于dup而言,返回的新文件描述符一定是当前可用的文件描述符中的最小数值,用dup2则可用用filedes参数指定新描述符的数值。如果filedes已经打开,则先将其关闭,如若filedes1等于filedes2,则dup2返回filedes2,而不关闭它。
//调用dup(oldfd)等效于
fcntl(oldfd,F_DUPFD,0); //即复制一个现有的描述符
上图显示了执行dup(1)后的内核数据结构。(内核使用三种数据结构表示打开的文件,注意三种数据结构之间的关系)
对于dup2函数,作用也是复制文件描述符,但是可指定文件描述符的值,见前面的注释。因为这个原因,作重定向时,通常可以按照如下方式应用:
dup2(fd,STDOUT_FILENO); //即将fd复制到STDOUT_FILENO,也就是说把输出指向fd所指向的文件。
另外再加一点,从shell中运行一个进程,默认会有3个文件描述符存在(0、1、2),0与进程的标准输入相关联,1与进程的标准输出相关联,2与进程的标准错误输出相关联。而我们的printf函数要想输出到屏幕上也需要STOUT_FIENO 而这时候它被指向了一个文件,这样经过这一步以后我们的printf就会直接输出到这个文件里而不会输出到屏幕上。但是有时候我们又需要我们写的程序输出到网页或者屏幕上调错,我们这时候就需要 fork()函数了。fork会复制一个和父进程状态一样的子进程,然后我们可以在子进程中执行dup2而父进程的输入输出不受影响。
下面看一道题目和一个例子:
题目:
解答:如果fd是1,前三句话的意思就是将文件描述符fd复制给0,1,2,只不过对于dup2(fd,1)而言,会返回1,但是没有关闭文件描述符1,此时3个文件描述符
指向相同的文件表项,所以并不会执行最后的判断;
如果fd是3,调用3次dup2后,会有4个文件描述符指向相同的文件表项:0,1,2,3,此时会关闭文件描述符3.
例子:
源程序:
#include <stdio.h>
#include <stdlib.h>
int main()
{
/*********************/
puts("hello!");//在标准输出上输出hello!
exit(0);
}
现需要只在星号上方加代码,使得输出的hello!在指定的文件中。例如:"/tmp/out"中》
#include <stdio.h>
#include <stdlib.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <unistd.h>
#define FNAME "/tmp/out"
int main()
{
int fd;
close(1);//关闭1文件描述符,依据文件描述符优先使用当前可用范围内最小的,所以按理后面的会占据1号
fd = open(FNAME,O_WRONLY | O_CREAT | O_TRUNC, 0600);
if(fd < 0)
{
perror("open()");
exit(1);
}
/*********************/
puts("hello!");//在标准输出上输出hello!
exit(0);
}
//加dup后
#include <stdio.h>
#include <stdlib.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <unistd.h>
#define FNAME "/tmp/out"
int main()
{
int fd;
//close(1);//关闭1文件描述符,依据文件描述符优先使用当前可用范围内最小的,所以按理后面的会占据1号
fd = open(FNAME,O_WRONLY | O_CREAT | O_TRUNC, 0600);
if(fd < 0)
{
perror("open()");
exit(1);
}
close(1);
dup(fd);//将fd复制一份,到当前可用的最小的文件描述符
close(fd);
/*********************/
puts("hello!");//在标准输出上输出hello!
exit(0);
}
//问题,如果当前进程fd就在1处,就会出现问题;并发执行时,可能还有其它线程打开了文件,先占用了1。 原因:close + dup操作不原子。 由dup2解决。
#include <stdio.h>
#include <stdlib.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <unistd.h>
#define FNAME "/tmp/out"
int main()
{
int fd;
//close(1);//关闭1文件描述符,依据文件描述符优先使用当前可用范围内最小的,所以按理后面的会占据1号
fd = open(FNAME,O_WRONLY | O_CREAT | O_TRUNC, 0600);
if(fd < 0)
{
perror("open()");
exit(1);
}
//close(1);
//dup(fd);//将fd复制一份,到当前可用的最小的文件描述符
dup2(fd,1);//它会首先close(1),然后将fd放到1号去,原子操作
close(fd);
/*********************/
puts("hello!");//在标准输出上输出hello!
exit(0);
}
//仍然有问题,如fd本身开始就是1怎么办,但dup2不会出现问题,看man dup2。问题在于后面的close(fd)。
#include <stdio.h>
#include <stdlib.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <unistd.h>
#define FNAME "/tmp/out"
int main()
{
int fd;
//close(1);//关闭1文件描述符,依据文件描述符优先使用当前可用范围内最小的,所以按理后面的会占据1号
fd = open(FNAME,O_WRONLY | O_CREAT | O_TRUNC, 0600);
if(fd < 0)
{
perror("open()");
exit(1);
}
//close(1);
//dup(fd);//将fd复制一份,到当前可用的最小的文件描述符
dup2(fd,1);//它会首先close(1),然后将fd放到1号去,原子操作。如果原fd是1,dup2也不会出现问题(看注释)。
if(fd != 1)
close(fd);
/*********************/
puts("hello!");//在标准输出上输出hello!
exit(0);
}