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 : %d\n", fdin);
printf("STDOUT fd : %d\n", 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):%d\n",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是否调用成功,出错处理也在程序中有体现。