APUE学习手札 编写一个与3.12节中dup2功能相同的函数,要求不调用fcntl函数,并且要有正确的出错处理

时间:2022-07-26
本文章向大家介绍APUE学习手札 编写一个与3.12节中dup2功能相同的函数,要求不调用fcntl函数,并且要有正确的出错处理,主要内容包括其使用实例、应用技巧、基本知识点总结和需要注意事项,具有一定的参考价值,需要的朋友可以参考一下。

3.2 编写一个与3.12节中dup2功能相同的函数,要求不调用fcntl函数,并且要有正确的出错处理。

思路,不断执行dup函数,直到返回与newfd相同的文件描述符,所有都执行结束之后关闭之前dup返回的文件描述符

不要忘记特判newfd和fd相同的情况,直接返回。记住dup2还多了一歩先关闭newfd的步骤

#include "apue.h"
#define BUFFSIZE 16
int main()
{
	char buffer[BUFFSIZE];
	int fdin,fdout,n;
	fdin=my_dup(STDIN_FILENO,3);
	fdout=my_dup(STDOUT_FILENO,4);
	if(fdin==-1||fdout==-1)
	{
		err_sys("my_dup error!");
		return -1;
	}
	else
	{
		printf("STDIN fd : %dn", fdin);
		printf("STDOUT fd : %dn", fdout);
		while((n=read(fdin,buffer,BUFFSIZE))>0)
		{
			if(write(fdout,buffer,n)!=n)
			{
				err_sys("write error!n");
			}
		}
		if(n < 0)printf("read error");
	}
	return 0;
}
int my_dup(int fd,int newfd)
{

	if(fd==newfd)return fd;
	if(fd<0||fd>FOPEN_MAX)
    {
        printf("fd is wrong.n");
        return -1;
    }
    if(newfd <0||newfd>FOPEN_MAX)
    {
        printf("newfd is wrong.n");
        return -1;
    }
	close(newfd);
	int fileindex[newfd+1];
	int index=0;
	while((fileindex[index++]=dup(fd))!=newfd)
	{
		printf("result after dup(fd):%dn",fileindex[index-1]);
		if(fileindex[index-1]==-1)
		{
			err_sys("my_dup error!");
			return -1;
		}
	}
	int i=0;
	for(;i<index-1;i++)
	{
		close(fileindex[i]);
	}
	return fileindex[index-1];
	
}

运行结果:

在服务器上编写3.2.c的源代码,编译,执行后如下图:

编译生成了一个3.2的执行文件,上述代码的功能是复制了STDIN_FILENO和STDOUT_FILENO这两个文件描述符,分别返回4和5

编译生成了一个3.2的执行文件,上述代码的功能是复制了STDIN_FILENO和STDOUT_FILENO这两个文件描述符,分别返回4和5

再通过读写验证my_dup是否调用成功,出错处理也在程序中有体现。