淺談dup和dup2的用法

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 的文件中。?
運行程序如下:?
這里寫圖片描述?
查看文件可以看到文件內容如下:?
這里寫圖片描述

  1. 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可以看到如下:?
這里寫圖片描述


本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/news/383668.shtml
繁體地址,請注明出處:http://hk.pswp.cn/news/383668.shtml
英文地址,請注明出處:http://en.pswp.cn/news/383668.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

C++ cin 實現循環讀入

習慣了使用while(~scanf("%d",x)){}來實現循環讀入&#xff0c;但是有時候使用泛型編程的時候就必須使用C中的cin&#xff0c;但是當我想要實現循環讀入的時候卻發現有些困難。 我們可以看一下下面這個簡單的例子&#xff1a; #include <iostream>using name…

BFPTR算法詳解+實現+復雜度證明

BFPTR算法是由Blum、Floyed、Pratt、Tarjan、Rivest這五位牛人一起提出來的&#xff0c;其特點在于可以以最壞復雜度為O(n)O(n)O(n)地求解top?ktop-ktop?k問題。所謂top?ktop-ktop?k問題就是從一個序列中求解其第k大的問題。 top?ktop-ktop?k問題有許多解決方法&#xff…

C++子類對象隱藏了父類的同名成員函數(隱藏篇)

https://blog.csdn.net/alpha_love/article/details/75222175#include <iostream>#include <stdlib.h>#include <string>using namespace std;/*** 定義人類: Person* 數據成員: m_strName* 成員函數: attack()*/class Person{public:Person(){cout<<&…

隨機化快速排序+快速選擇 復雜度證明+運行測試

對于快速排序和快速選擇我之前的文章已經有詳細的說明&#xff0c;需要了解的同學可以移步 傳送門&#xff1a;快速排序&#xff5c;快速選擇(BFPTR) 所謂隨機化其實就是選擇樞紐的時候使用隨機數選擇而已&#xff0c;實現起來很簡單。但是我們使用隨機數如何保證復雜度呢&am…

C++子類父類成員函數的覆蓋和隱藏實例詳解

https://www.jb51.net/article/117380.htm函數的覆蓋覆蓋發生的條件&#xff1a; &#xff08;1&#xff09; 基類必須是虛函數&#xff08;使用virtual 關鍵字來進行聲明&#xff09; &#xff08;2&#xff09;發生覆蓋的兩個函數分別位于派生類和基類 &#xff08;3&#xf…

【Linux基礎】Linux的5種IO模型詳解

引入 為了更好的理解5種IO模型的區別&#xff0c;在介紹IO模型之前&#xff0c;我先介紹幾個概念 1.進程的切換 &#xff08;1&#xff09;定義 為了控制進程的執行&#xff0c;內核必須有能力掛起正在CPU上運行的進程&#xff0c;并恢復以前掛起的某個進程的執行。即從用戶…

計算機網絡【五】廣播通信+以太網

局域網的拓撲 廣域網使用點到點通信 局域網使用廣播通信 可以隨意向網絡中添加設備。 總線網星形網&#xff0c;使用集線器。現在多使用星形網絡。環狀網樹形網 其中匹配電阻用來吸收總線上傳播的信號。 共享通信媒體 靜態劃分信道 頻分復用、時分復用、波分復用、碼分復用…

聊聊Linux 五種IO模型

一篇《聊聊同步、異步、阻塞與非阻塞》已經通俗的講解了&#xff0c;要理解同步、異步、阻塞與非阻塞重要的兩個概念點了&#xff0c;沒有看過的&#xff0c;建議先看這篇博文理解這兩個概念點。在認知上&#xff0c;建立統一的模型。這樣&#xff0c;大家在繼續看本篇時&#…

操作系統【四】分頁存儲管理

連續分配方式的缺點&#xff1a; 固定分區分配&#xff1a;缺乏靈活性&#xff0c;產生大量的內部碎片&#xff0c;內存的利用率較低 動態分區分配&#xff1a;會產生許多外部碎片&#xff0c;雖然可以用緊湊技術處理&#xff0c;但是緊湊技術的時間代價較高 基本分頁存儲管理…

聊聊同步、異步、阻塞與非阻塞

近來遇到了一些常見的概念&#xff0c;尤其是網絡編程方面的概念&#xff0c;如&#xff1a;阻塞、非阻塞、異步I/O等等&#xff0c;對于這些概念自己也沒有太清晰的認識&#xff0c;只是很模糊的概念&#xff0c;說了解吧也了解&#xff0c;但是要讓自己準確的描述概念方面的具…

操作系統【五】分段內存管理+段頁式內存管理

基本分段存儲管理 與分頁最大的區別&#xff1a;離散分配時所分配地址空間的基本單位不同 進程的地址空間&#xff1a;按照程序自身的邏輯關系劃分為若干個段&#xff0c;每個段都有一個段名&#xff0c;每段從0開始編址 內存分配規則&#xff1a;以段位單位進行分配&#xff…

計算機網絡【六】網絡層協議

網絡層負責在不同網絡之間盡力轉發數據包&#xff08;基于數據包的IP地址轉發&#xff09;。不負責丟失重傳&#xff0c;也不負責順序&#xff08;每一個數據包都是單獨選擇路徑&#xff09;。 可靠傳輸是由傳輸層實現。 網絡設備和OSI參考模型 通過分層&#xff0c;屏蔽了…

epoll 水平觸發與邊緣觸發

https://blog.csdn.net/lihao21/article/details/67631516?refmyread epoll也是實現I/O多路復用的一種方法&#xff0c;為了深入了解epoll的原理&#xff0c;我們先來看下epoll水平觸發&#xff08;level trigger&#xff0c;LT&#xff0c;LT為epoll的默認工作模式&#xff…

計算機網絡【3】網絡層

主要任務時把分組從源端發送到目的端&#xff0c;為分組交換網上的不同主機提供服務。網絡層傳輸單位是數據報 功能&#xff1a; 路由選擇與分組轉發&#xff08;最佳路徑 &#xff09;異構網絡互聯擁塞控制 數據交換方式 電路交換&#xff1a;通信時延小、有序傳輸、沒有沖…

C++空類的大小

https://blog.csdn.net/lihao21/article/details/47973609 本文中所說是C的空類是指這個類不帶任何數據&#xff0c;即類中沒有非靜態(non-static)數據成員變量&#xff0c;沒有虛函數(virtual function)&#xff0c;也沒有虛基類(virtual base class)。 直觀地看&#xff0c…

Linux探秘之用戶態與內核態

https://www.cnblogs.com/bakari/p/5520860.html 一、 Unix/Linux的體系架構 如上圖所示&#xff0c;從宏觀上來看&#xff0c;Linux操作系統的體系架構分為用戶態和內核態&#xff08;或者用戶空間和內核&#xff09;。內核從本質上看是一種軟件——控制計算機的硬件資源&…

哈夫曼算法證明+哈夫曼編碼譯碼程序實現

哈夫曼算法證明 哈夫曼算法是一種貪心算法&#xff0c;我們考慮證明其最優子結構和貪心選擇性質&#xff1a; 最優子結構&#xff1a;假設一個樹是哈夫曼樹&#xff0c;則以其任意節點為根節點的最大子樹也是哈夫曼樹。 證明&#xff1a;子樹的根節點的值是其所有葉子節點出現…

Python3小知識

對于迭代器對象&#xff0c;Python默認賦值是將引用賦值&#xff0c;即指向同一片內存空間。為了實現對內存空間的賦值&#xff0c;我們可以使用分片進行深復制。例如&#xff1a; 當定義元組的時候&#xff0c;我們一般使用小括號將元素包圍起來&#xff0c;也可以不使用括號…

匯編:實現日歷星期數查詢工具

編制一個簡單日歷查詢工具&#xff0c;輸入年、月、日&#xff0c;能夠判斷當日的星期數&#xff0c;并進行輸出&#xff0c;數據的輸入和結果的輸出要有必要的提示&#xff0c;且提示獨占一行。 查閱資料 ? 經過查閱資料&#xff0c;發現有兩個相關的算法可以解決這個問題&…

一個通用純C隊列的實現

https://blog.csdn.net/kxcfzyk/article/details/31728179 隊列并不是很復雜的數據結構&#xff0c;但是非常實用&#xff0c;這里實現一個隊列是因為在我的另一篇博客非常精簡的Linux線程池實現中要用到。 隊列API定義如下&#xff1a; //queue.h #ifndef QUEUE_H_INCLUDED…