Unix环境高级编程学习笔记——dup

时间:2022-04-17 00:10:31

dup 和 dup2
    dup和dup2,都是用来将一个文件描述符复制给另一个文件描述符上,这两个文件描述符都指向同一个文件状态标志上。
  只是文件描述符的大小不一样,dup所执行下的复制,肯定是返回的系统中最小的可用文件描述符。而dup2.可以自定义文件描述符的。系统的0、1、2描
  述符已经被占用。所以用dup所复制的描述符是不能是这三个的。我们可以将这三个复制给其他的文件描述符。

函数描述

  int dup(int filedes)

  int dup2(int filedes, int filedes2)

dup

   dup的参数是filedes,这个指的是系统中已经打开的文件描述符.返回的确实一个可用的最小文件描述符。

  利用fcntl,我们可以将其分解看:fcntl(filedes ,F_DUPFD,0)

   可以很简单的看一下这三个参数,第一是我们要复制的文件描述符,第二个是fcntl的状态标志,指的是dup一个文件描述符。第三个参数,我们很容易理解错,这边写的是0,有可能你会认为是将0号描述符给返回出去,其实不是,他的是意思 是,返   回一个比这个数大或是等于他的最小描述符。这个和dup的功能解释是符合的。

dup2

  他的参数是有俩,filedes和前面的一样,是我们要复制的文件描述符,后面的filedes2,则是我们想要函数返回的文件描述符,很好理解。

   同样这fcntl中我们可以很清晰的看出:fcntl(filedes, F_DUPFD,filedes2).

  先前的“0”,被我们用filedes2直接替代了。函数返回的就是我们设置的那个filedes2。

confuse

  dup2,对于这个函数,我们可以自己定义文件描述符,这就有一个容易混淆的地方,如果我们将fd定义为从STDIN_FILENO复制过来的,现在我们用fd2 = dup2(fd,1)。我们知道文件描述符1已经是被标准输出占用了,你会说我们现在是在将标准输入   复制给“标准输出”,先不想这个!按照原先的思路现在我们用的fd2应该就是一个标准输入,我们可以拿他来从屏幕上读一个串数据,然后显示到终端上,这个是可以做到的。但是你会说这个fd2不是被赋予了1的描述符吗?怎么可以用作标准输入    呢?回到dup2的定义,dup2在执行的过程中,会看这个filedes2是不是已经打开? 如果打开了,就将其关闭。然后再执行复制,也就是说,这边的fd2就是指向的标准输入了。一个描述符只能指向一个文件状态标志的。所以这个是不冲突的。而标  准输入已经被关闭了。下面是这个confuse验证代码:

 #include<stdio.h>
#include<stdlib.h>
#include<string.h>
#include<unistd.h>
#include<fcntl.h>
#include<errno.h> #define BUFSIZE 4096 int main(void)
{
int fd;
int fd2,n;
char buf[BUFSIZE]; fd = dup(STDIN_FILENO);
if(fd < )
{
fprintf(stderr,"dup:%s\n",strerror(errno));
exit();
}
fd2 = dup2(fd,);
if(fd2<)
{
fprintf(stderr,"dup2:%s\n",strerror(errno));
exit();
} printf("fd = %d \n",fd);
printf("fd2 = %d \n",fd2); if((n =read(fd,buf,BUFSIZE)) < )
{
fprintf(stderr,"read:%s\n",strerror(errno));
exit();
} if(write(fd2,buf,n) != n)
{
fprintf(stderr,"write:%s\n",strerror(errno));
exit();
} close(fd);
close(fd2);
return ;
}

结果截图:Unix环境高级编程学习笔记——dup

My_dup2

  要求: 要实现自己写一个dup2,不使用fcntl函数,有正确的错误提示。

  分析: 不使用fcntl函数,那就只有使用dup来实现这个dup2了。我们的目的就是让自己所定义的描述符能过符合要求,系统有个要求,最大的描述符不能超过63,也就是最多打开64个描述符,这是第一个要检查的。继而我们检查,这个所定义的描    述符,和我要复制的这个描述符是不是相等的,要是相等的话,那就直接返回了。这是第二个要检查的。好了,这些都符合要求,那么我们就定义一个额外的文件描述符(区别于你给定的文件描述符)。我们用这个额外的一个描述符来接受我们对    我们要复制的描述符的dup的返回值。将这个返回值和你所给定的比对,如果相等就将这个fd返回,说明已经将其复制好。也就完成了dup2的工作了。 解释一下为什么要这么做: dup的返回值是系统中最小的可用的文件描述符,所以我们不能    够一下子就能确定dup返回的这个文件描述符就是你所指定的,同时dup操作也是在将你说指定文件描述符指向你所要复制的那个文件描述符指向的文件状态标志上去(实际上不是,实际上同过dup不断打开的文件描述符,将在你所指定的文件描    述符前的文件描述符占用掉,*你的返回你所指定的文件描述符。所以我们要将这些之前不是我们所需的文件描述符保存到一个数组中,待最后找到指定的描述符后就将一一释放掉。)

下面的一个给出可行的代码:

 /************************************************************************************************************
*程序目的是通过dup实现dup2,并用通过dup2所返回的文件描述符实现文件的写操作
*************************************************************************************************************/
#include<stdio.h>
#include<stdlib.h>
#include<string.h>
#include<unistd.h>
#include<fcntl.h>
#include<errno.h> //系统在fcntl这个头文件里应该有OPEN_MAX这个参数的,不过我的系统上引入头文件也是没用的
//故用条件编译的原理让其自动判断并定义
#ifndef OPEN_MAX
#define OPEN_MAX 256
#endif char buf1[] = "abcdefg";
char buf2[] = "ABCDEFG"; int mydup2(int ,int); int main()
{
int fd,fd2; fd = open("a.txt",O_RDWR|O_CREAT,);
if(fd < )
{
fprintf(stderr, "open:%s\n",strerror(errno));
exit();
} if(write(fd,buf1,strlen(buf1)) != strlen(buf1))
{
fprintf(stderr,"write:%s\n",strerror(errno));
exit();
}
//实现文件描述符的复制
fd2 = mydup2(fd,); if(write(fd2,buf2,strlen(buf2)) != strlen(buf2))
{
fprintf(stderr,"write2:%s\n",strerror(errno));
exit();
} printf("fd2:%d\n",fd2); return ; } int mydup2(int filedes,int filedes2)
{
int fd[OPEN_MAX];
int i =-;
int tfd;
//检查你指定的文件描述符是否合法
if(filedes < || filedes2 < || filedes2 > OPEN_MAX)
{
printf("filedes is error!\n");
exit();
} if(filedes == filedes2)
{
return filedes2;
}
close(filedes2);
//这边是遍历寻找你所指定的文件描述符
tfd = dup(filedes);
while(filedes2 != tfd)
{
i++;
fd[i] = tfd;
tfd = dup(filedes);
}
//释放之前打开的没有用的文件描述符
while(i>=)
{
close(fd[i]);
i--;
}
return filedes2;
}

运行结果:Unix环境高级编程学习笔记——dup

上述也就是对dup和dup2的一些理解。谷歌了半天发现这类的资料写的都比较官方,自己看不大懂,自己写下自己的一些理解,也是算是自己的一个学习的验证了。