关于dup和dup2函数

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);
}

参考:
关于dup和dup2
APUE习题解答
dup和dup2

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值