【Linux】有名管道實現Linux進程間通信

来源:https://www.cnblogs.com/Zoya-/archive/2022/09/23/16724617.html
-Advertisement-
Play Games

1.加拿大創新、科學和經濟發展部 (ISED) 於 2022 年 9 月 9 日發佈了第 2022-CEB001 號通知。 該通知包括關於無線電標準規範 RSS-195 “無線通信服務 (WCS) 設備在 2305-2320 MHz 和 2345-2360 MHz 頻段”第 2 版的指南,旨在重申 ...


進程間通信之有名管道​

進程間通信有多種方式實現,本文主要講解有名管道的通信方式。

一, 有名管道簡介

匿名管道由於沒有名字,只能用於具有親緣關係的進程間通信。

為了剋服這個缺點,就提出了有名管道(FIFO),也稱為命名管道FIFO文件

有名管道(FIFO)提供了一個路徑名與之關聯,以FIFO的文件形式存在於文件系統中,且打開方式與打開一個普通文件是一樣的,即使與FIFO的創建進程不存在親緣關係的進程,只要可以訪問該路徑,就能夠彼此通過FIFO相互通信。因此,通過FIFO不相關的進程也能交換數據。

FIFO文件被打開,就可以使用與操作匿名管道和其它文件的系統調用一樣的I/O系統調用,如使用 read()讀數據write()寫數據close()關閉FIFO等。

與管道一樣,FIFO也有一個寫入端和讀取端,且從管道中讀取數據的順序與寫入數據的順序是一樣的。FIFO的名稱也由此而來:先入先出。也是一個環形隊列。

有名管道和匿名管道大部分是相同的,不同在於:

  • FIFO在文件系統中作為一個特殊文件存在,FIFO的內容存放在記憶體中;
  • 當使用FIFO的進程退出後,FIFO文件將繼續保存在文件系統中以便以後使用;

二, 有名管道的使用

1. 創建有名管道

  • 方式1:可以使用命令創建有名管道:
mkfifo 名字
  • 方式2:使用函數mkfifo()函數創建有名管道

使用mkfifo創建了FIFO後,就可以使用open打開它,常見的文件I/O函數都可以用於FIFO。

FIFO嚴格遵循先進先出,對FIFO的讀總是從開始處返回數據,對FIFO的寫則是把數據添加到末尾,FIFO不支持lseek()等文件定位的函數。

函數mkkfifo()聲明:

#include <sys/types.h>
#include <sys/stat.h>
int mkfifo(const char *pathname, mode_t mode);
  • 參數說明:
    • pathname:管道名稱的路徑;
    • mode:FIFO的許可權,和open是一樣的;如0664;
  • 返回值:成功返回0,失敗返回-1,並設置對應的errno;

創建管道示例:

#include <sys/types.h>
#include <sys/stat.h>
#include <stdio.h>

int main()
{
    // 創建有名管道
    int ret = mkfifo("fifo1",0664);
    if(ret == -1)
    {
        perror("mkfifo");
        return -1;
    }

    return 0;
}

有名管道使用示例,有兩個文件,read.c用來讀管道中的數據,write.c用來向管道寫數據。read.c和write.c的內容如下:

read.c

// 從管道中讀取數據
#include <sys/types.h>
#include <sys/stat.h>
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <fcntl.h>

int main()
{

    // 1. 打開管道文件
    int fd = open("test",O_RDONLY);  // 阻塞,如果沒有寫端打開那麼會一直阻塞在這裡
    if(fd == -1){
        perror("open");
        exit(0);
    }

    // 2. 讀數據
    while (1)
    {
        char buf[1024]={0};
        int len = read(fd,buf,sizeof(buf));
        if(len == -1){
            perror("read");
            exit(0);
        }
        else if(len == 0){
            // len = 0表示已經讀到管道末尾,寫端斷開連接了
            break;
        }
        printf("receive buf: %s\n",buf);
    }

    // 3. 關閉FIFO
    close(fd);
    

    return 0;
}

write.c:

// 向管道中寫數據
#include <sys/types.h>
#include <sys/stat.h>
#include <stdio.h>
#include <unistd.h>
#include <fcntl.h>
#include <string.h>
#include <stdlib.h>

int main()
{
    // 創建有名管道
    // 1. 先判斷文件是否存在
    if(-1 == access("test",F_OK))   // 判斷文件是否存在
    {
        printf("管道不存在,創建管道\n");
        // 2. 創建管道文件
        int ret = mkfifo("test",0664);
        if(ret == -1)
        {
            perror("mkfifo");
            return -1;
        }
    }

    // 3. 打開管道,以只寫方式打開管道
    int fd = open("test",O_WRONLY);  // 阻塞方式,如果沒有讀端打開,那麼會一直阻塞在這裡
    if(fd == -1)
    {
        perror("open");
        exit(0);
    }

    // 4. 向管道中寫入數據
    for(int i=0;i<100;i++){
        char buf[1024]={0};
        sprintf(buf,"hello, %d\n",i);
        printf("write data : %s\n",buf);
        write(fd,buf,strlen(buf));
        sleep(1);
    }

    // 5. 關閉FIFO
    close(fd);

    return 0;
}

生成可執行文件 read 和 write,當只執行read或只執行write時,會一直阻塞;當兩個文件都執行時,會進行數據傳遞;

2. 有名管道的註意事項

  • 一個進程以只讀打開管道會阻塞,直到另外一個進程以只寫打開管道;
  • 一個進程以只寫打開管道會阻塞,直到另外一個進程以只讀打開管道;

有名管道的讀寫特性:

  • 讀管道
    • 管道中有數據,read返回實際讀到的位元組數;
    • 管道中無數據:
      • 管道寫端被全部關閉,read返回0,相當於讀到文件末尾;
      • 管道寫端沒有被全部,read阻塞等待;
  • 寫管道
    • 管道讀端全部關閉,進行異常終止,收到信號SIGPIPE;
    • 管道讀端沒有被全部關閉:
      • 管道已經滿了,write阻塞等待
      • 管道沒有滿,write將數據寫入,並返回實際寫入的實際位元組數


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 我的博客 一直有個想法,想要弄個微信機器人,然而出師不利,剛開始就碰壁了 先上代碼,這個是用來接收消息的,是個測試腳本 #!/usr/bin/python # coding: utf-8 import itchat def write_infomation(text_value): print(te ...
  • WPS表格文件是金山開發的專門用於處理表格數據的Office工具,屬於WPS Office中WPS文字、WPS表格和WPS演示三大功能模塊之一。通常以.et和.ett作為文件尾碼。我們在通過後端來操作WPS表格文件時,可以通過以下方法來載入、編輯以及保存WPS表格文件,本文將對此做詳細介紹。 引入j ...
  • 1 #include <stdio.h> 2 #include <malloc.h> 3 #include <stdlib.h> 4 #define MaxSize 50 5 #define InitSize 100 6 typedef int ElemType; 7 typedef struct ...
  • 問題描述:新創建maven項目後,在父工程中dependencyManagement時,會報紅線錯誤,刷新後還是報紅,例如:${spring.version}爆紅: Maven使用dependencyManagement元素來進行依賴版本的管理。具體來說,maven沿著父子層向上尋找,直到找到dep ...
  • 在上一篇:UWP/WinUI3 PixelShaderEffect 實現ThresholdEffect 濾鏡。 - 吃飯/睡覺 - 博客園 (cnblogs.com) 已經價紹瞭如何編寫hsls,編譯,和使用 PixelShaderEffect 來實現自定義濾鏡效果了,那麼本編將介紹如何編寫一個 “ ...
  • 在上一個文章中,傳送門,給大家介紹了怎麼在配置文件中使用 Kestrel 部署 Https,正好今天有小伙伴穩問到:可以通過代碼的方式實現 Kestrel 的 Https 的部署嗎?答案是肯定的,我們這次一樣去不是多個功能變數名稱。 在使用代碼實現中,我是主要使用到 ListenOptions.UseHtt ...
  • 1. 融合效果 在 CSS 中有一種實現融合效果的技巧,使用模糊濾鏡(blur)疊加對比度濾鏡(contrast)使兩個接近的元素看上去“粘”在一起,如下圖所示: 博客園的 ChokCoco 就用這個技巧實現了很多不同的玩法並寫了很多文章,例如這篇: 你所不知道的 CSS 濾鏡技巧與細節 我一直對這 ...
  • zookeeper ##協調機制 選舉leader 多個flower 客戶端 伺服器 ##特點 半數以上 數據一致性 在有限時間範圍內,執行順序同步於發送順序 文件結構類unix 樹狀每一個結點既是文件夾也可以是值。記為znode ? 本質上zookeeper 是文件系統+通知機制 ##啟動zook ...
一周排行
    -Advertisement-
    Play Games
  • GoF之工廠模式 @目錄GoF之工廠模式每博一文案1. 簡單說明“23種設計模式”1.2 介紹工廠模式的三種形態1.3 簡單工廠模式(靜態工廠模式)1.3.1 簡單工廠模式的優缺點:1.4 工廠方法模式1.4.1 工廠方法模式的優缺點:1.5 抽象工廠模式1.6 抽象工廠模式的優缺點:2. 總結:3 ...
  • 新改進提供的Taurus Rpc 功能,可以簡化微服務間的調用,同時可以不用再手動輸出模塊名稱,或調用路徑,包括負載均衡,這一切,由框架實現並提供了。新的Taurus Rpc 功能,將使得服務間的調用,更加輕鬆、簡約、高效。 ...
  • 本章將和大家分享ES的數據同步方案和ES集群相關知識。廢話不多說,下麵我們直接進入主題。 一、ES數據同步 1、數據同步問題 Elasticsearch中的酒店數據來自於mysql資料庫,因此mysql數據發生改變時,Elasticsearch也必須跟著改變,這個就是Elasticsearch與my ...
  • 引言 在我們之前的文章中介紹過使用Bogus生成模擬測試數據,今天來講解一下功能更加強大自動生成測試數據的工具的庫"AutoFixture"。 什麼是AutoFixture? AutoFixture 是一個針對 .NET 的開源庫,旨在最大程度地減少單元測試中的“安排(Arrange)”階段,以提高 ...
  • 經過前面幾個部分學習,相信學過的同學已經能夠掌握 .NET Emit 這種中間語言,並能使得它來編寫一些應用,以提高程式的性能。隨著 IL 指令篇的結束,本系列也已經接近尾聲,在這接近結束的最後,會提供幾個可供直接使用的示例,以供大伙分析或使用在項目中。 ...
  • 當從不同來源導入Excel數據時,可能存在重覆的記錄。為了確保數據的準確性,通常需要刪除這些重覆的行。手動查找並刪除可能會非常耗費時間,而通過編程腳本則可以實現在短時間內處理大量數據。本文將提供一個使用C# 快速查找並刪除Excel重覆項的免費解決方案。 以下是實現步驟: 1. 首先安裝免費.NET ...
  • C++ 異常處理 C++ 異常處理機制允許程式在運行時處理錯誤或意外情況。它提供了捕獲和處理錯誤的一種結構化方式,使程式更加健壯和可靠。 異常處理的基本概念: 異常: 程式在運行時發生的錯誤或意外情況。 拋出異常: 使用 throw 關鍵字將異常傳遞給調用堆棧。 捕獲異常: 使用 try-catch ...
  • 優秀且經驗豐富的Java開發人員的特征之一是對API的廣泛瞭解,包括JDK和第三方庫。 我花了很多時間來學習API,尤其是在閱讀了Effective Java 3rd Edition之後 ,Joshua Bloch建議在Java 3rd Edition中使用現有的API進行開發,而不是為常見的東西編 ...
  • 框架 · 使用laravel框架,原因:tp的框架路由和orm沒有laravel好用 · 使用強制路由,方便介面多時,分多版本,分文件夾等操作 介面 · 介面開發註意欄位類型,欄位是int,查詢成功失敗都要返回int(對接java等強類型語言方便) · 查詢介面用GET、其他用POST 代碼 · 所 ...
  • 正文 下午找企業的人去鎮上做貸後。 車上聽同事跟那個司機對罵,火星子都快出來了。司機跟那同事更熟一些,連我在內一共就三個人,同事那一手指桑罵槐給我都聽愣了。司機也是老社會人了,馬上聽出來了,為那個無辜的企業經辦人辯護,實際上是為自己辯護。 “這個事情你不能怪企業。”“但他們總不能讓銀行的人全權負責, ...