dup 与 dup2

1. 文件描述符在内核中数据结构

 一个进程在此存在期间,会有一些文件被打开,从而会返回一些文件描述符,从shell

中运行一个进程,默认会有3个文件描述符存在(0、1、2),

0与进程的标准输入相关联,

1与进程的标准输出相关联,

2与进程的标准错误输出相关联,

一个进程当前有哪些打开的文件描述符可以通过/proc/进程ID/fd目录查看。

 下图可以清楚的说明问题:

  进程表项
————————————————

   fd标志 文件指针
      _____________________
fd 0:|________|____________|————> 文件表
fd 1:|________|____________|
fd 2:|________|____________|
fd 3:|________|____________|
       |     …….         |
       |_____________________|

                图1
       
文件表中包含:文件状态标志、当前文件偏移量、v节点指针,这些不是本文讨论的

重点,我们只需要知道每个打开的文件描述符(fd标志)在进程表中都有自己的文件表

项,由文件指针指向。

2. dup/dup2函数

APUE和man文档都用一句话简明的说出了这两个函数的作用:复制一个现存的文件描述符。

#include <unistd.h>

int dup(int oldfd);

int dup2(int oldfd, int newfd);

从图1来分析这个过程,当调用dup函数时,内核在进程中创建一个新的文件描述符,此

描述符是当前可用文件描述符的最小数值,这个文件描述符指向oldfd所拥有的文件表项。

  进程表项
————————————————

   fd标志 文件指针
      _____________________
fd 0:|________|____________|                          ______
fd 1:|________|____________|—————-> |               |
fd 2:|________|____________|                       | 文件表 |
fd 3:|________|____________|—————-> | _____  |
       |     …….         |
       |_____________________|

                图2:调用dup后的示意图

如图2 所示,假如oldfd的值为1, 当前文件描述符的最小值为3, 那么新描述符3指向

描述符1所拥有的文件表项。

dup2和dup的区别就是可以用newfd参数指定新描述符的数值,如果newfd已经打开,则

先将其关闭。如果newfd等于oldfd,则dup2返回newfd, 而不关闭它。dup2函数返回的新

文件描述符同样与参数oldfd共享同一文件表项。

APUE用另外一个种方法说明了这个问题:

实际上,调用dup(oldfd);

等效与
        fcntl(oldfd, F_DUPFD, 0)

而调用dup2(oldfd, newfd);

等效与
        close(oldfd);
        fcntl(oldfd, F_DUPFD, newfd); 

[cpp] view plain copy print ?
  1. #include <sys/stat.h>   
  2. #include <string.h>   
  3. #include <fcntl.h>   
  4. #include <stdio.h>   
  5. #include <unistd.h>   
  6. int main(void)  
  7. {  
  8.    #define STDOUT 1   //标准输出文件描述符 号   
  9.    int nul, oldstdout;  
  10.    char msg[] = "This is a test";  
  11.    //打开一个文件,操作者具有读写权限 如果文件不存在就创建   
  12.    nul = open("DUMMY.FIL", O_CREAT | O_RDWR, S_IREAD | S_IWRITE);  
  13.    /* create a duplicate handle for standard 
  14.       output */  
  15.    oldstdout = dup(STDOUT);  
  16.    /* 
  17.       redirect standard output to DUMMY.FIL 
  18.       by duplicating the file handle onto the 
  19.       file handle for standard output. 
  20.    */  
  21.    dup2(nul, STDOUT);  
  22.    /* close the handle for DUMMY.FIL */  
  23.    close(nul);  
  24.    /* will be redirected into DUMMY.FIL */  
  25.    write(STDOUT, msg, strlen(msg));  
  26.    /* restore original standard output 
  27.       handle */  
  28.    dup2(oldstdout, STDOUT);  
  29.    /* close duplicate handle for STDOUT */  
  30.    close(oldstdout);  
  31.    return 0;  
  32. }  

你可能感兴趣的:(数据结构,shell,文档)