https://blog.csdn.net/u012058778/article/details/78705536
一、dup和dup2函數?
這兩個函數都可以來復制一個現有的文件描述符,他們的聲明如下:
#include <unistd.h>int dup(int fd);int dup2(int fd, int fd 2);
- 1
- 2
- 3
關于dup函數,當我們調用它的時候,dup會返回一個新的描述符,這個描述一定是當前可用文件描述符中的最小值。我們知道,一般的0,1,2描述符分別被標準輸入、輸出、錯誤占用,所以在程序中如果close掉標準輸出1后,調用dup函數,此時返回的描述符就是1。?
對于dup2,可以用fd2指定新描述符的值,如果fd2本身已經打開了,則會先將其關閉。如果fd等于fd2,則返回fd2,并不關閉它。?
這兩個函數返回的描述符與fd描述符所指向的文件共享同一文件表項。如下圖所示:??
也就是fd與fd2可對同一個文件進行讀寫操作。且其是一種原子操作。?
二、重定向示例?
1. dup
8 #include <stdio.h>9 #include <unistd.h>10 #include <stdlib.h>11 #include <sys/stat.h>12 #include <fcntl.h>13 14 int main(int argc, char* argv[])15 {16 int i_fd = open("hello.txt", O_CREAT|O_APPEND|O_RDWR, 0666);17 18 if(i_fd < 0)19 {20 printf("open error!\n");21 return 0;22 }23 24 if(write(i_fd, "hello fd\n", 9) != 9)25 {26 printf("write fd error\n");27 28 }29 30 int i_dup_fd = dup(i_fd);31 if(i_dup_fd < 0)32 {33 printf("dup error!\n");34 return 0;35 }36 37 printf("i_dup_fd = %d \t i_fd = %d\n", i_dup_fd, i_fd);38 close(i_fd);39 40 char c_buffer[100];41 int n = 0;42 while((n = read(STDIN_FILENO, c_buffer, 1000)) != 0)43 {44 if(write(i_dup_fd, c_buffer, n) != n)45 {46 printf("write dup fd error!\n");47 return 0;48 }49 }50 return 0;51 }
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
- 10
- 11
- 12
- 13
- 14
- 15
- 16
- 17
- 18
- 19
- 20
- 21
- 22
- 23
- 24
- 25
- 26
- 27
- 28
- 29
- 30
- 31
- 32
- 33
- 34
- 35
- 36
- 37
- 38
- 39
- 40
- 41
- 42
- 43
- 44
如上打開一個文件,我們先寫入文件內容“hello fd”,然后將fd的描述符拷貝到dup_fd 的文件描述符上,然后將標準輸入的內容寫入到dup_fd 的文件中。?
運行程序如下:??
查看文件可以看到文件內容如下:?
- dup2?
如下關于dup2的使用:
8 #include <stdio.h>9 #include <stdlib.h>10 #include <string.h>11 #include <unistd.h>12 #include <sys/stat.h>13 #include <fcntl.h>14 15 int main(int argc, char* argv[])16 {17 int i_fd = open("hello_dup2.txt", O_CREAT|O_APPEND|O_RDWR, 0666);18 19 20 if(i_fd < 0)21 {22 printf("open error!\n");23 return 0;24 }25 26 if(write(i_fd, "hello i_fd\n", 11) != 11)27 {28 printf("write dup2 error\n");29 }30 31 32 int i_dup2_fd = dup2(i_fd, STDOUT_FILENO);33 34 if(i_dup2_fd != STDOUT_FILENO)35 {36 printf("error dup2!\n");37 return 0;38 }39 close(i_fd);40 41 char c_buf[1024];42 int i_read_n = 0;43 while((i_read_n = read(STDIN_FILENO, c_buf, 1024)) != 0)44 {45 i_read_n = read(STDIN_FILENO, c_buf + i_read_n, sizeof(c_buf) - 1 - i_read_n);46 47 if(i_read_n < 0)48 {49 printf("read error!\n");50 return 0;51 }52 53 printf("%s", c_buf);54 fflush(stdout);55 sleep(1);56 }57 close(i_dup2_fd);58 59 return 0;60 61 }
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
- 10
- 11
- 12
- 13
- 14
- 15
- 16
- 17
- 18
- 19
- 20
- 21
- 22
- 23
- 24
- 25
- 26
- 27
- 28
- 29
- 30
- 31
- 32
- 33
- 34
- 35
- 36
- 37
- 38
- 39
- 40
- 41
- 42
- 43
- 44
- 45
- 46
- 47
- 48
- 49
- 50
- 51
- 52
- 53
- 54
如上,這里沒有像使用dup的時候顯示的調用write函數將標準輸入的內容寫入到指定文件中,而是將標準輸出重定向到指定文件中,然后調用printf函數將標準輸出的內容重定向到指定文件中。我們在寫簡單的日志時就可以將printf的內容重定向到日志中,使用printf作為寫日志的接口。?
如上運行程序如下:?
查看hello_dup2.txt可以看到如下:?