操作系統第三次實驗報告——有名管道(FIFO)

来源:https://www.cnblogs.com/MilkoSilver/archive/2020/04/15/12704845.html
-Advertisement-
Play Games

0 個人信息 張櫻姿 201821121038 計算1812 1 實驗目的 掌握進程間通信管道的編程。 2 實驗內容 在伺服器上用VIM編寫一個程式:創建一個命名管道,創建兩個進程分別對管道進行讀(read_fifo.c)和寫(write_fifo.c)。給出源代碼 給出運行結果,並分析 3 實驗報 ...


0 個人信息

  • 張櫻姿
  • 201821121038
  • 計算1812

1 實驗目的

  • 掌握進程間通信管道的編程。

2 實驗內容

  • 在伺服器上用VIM編寫一個程式:創建一個命名管道,創建兩個進程分別對管道進行讀(read_fifo.c)和寫(write_fifo.c)。給出源代碼
  • 給出運行結果,並分析

3 實驗報告

  3.1 編寫寫管道程式(write_fifo.c)

 1 #include<unistd.h>    //write,read,close,access
 2 #include<string.h>    //memset
 3 #include<errno.h>     //errno
 4 #include<fcntl.h>     //open,O_WRONLY,O_RDONLY
 5 #include<stdio.h>     //printf,sscanf
 6 #include<stdlib.h>    //exit
 7 #include<limits.h>    //PIPE_BUF
 8 
 9 #define MYFIFO "/tmp/myfifo" //有名管道文件名
10 #define BUFES PIPE_BUF
11 
12 int main(int argc,char * argv[])
13 {
14     int fd,n;
15     char buff[BUFES];
16     
17     if(argc <= 1)
18     {
19         exit(1);
20     }
21     sscanf(argv[1],"%s",buff);
22     //以只寫阻塞方式打開FIFO管道
23     fd = open(MYFIFO,O_WRONLY);
24     if(fd==-1)
25     {
26         printf("Open fifo error\n");
27         exit(1);
28     }
29     //向管道中寫入字元串
30     if((n = write(fd,buff,BUFES))>0)
31     {
32         printf("Finish writing '%s' to FIFO\n",buff);
33     }
34     close(fd);
35     exit(0);
36 }

  3.2 編寫讀管道程式(read_fifo.c)

 1 #include<unistd.h>    //write,read,close,access
 2 #include<string.h>    //memset
 3 #include<errno.h>     //errno
 4 #include<fcntl.h>     //open,O_WRONLY,O_RDONLY
 5 #include<stdio.h>     //printf,sscanf
 6 #include<stdlib.h>    //exit
 7 #include<limits.h>    //PIPE_BUF
 8 
 9 #define MYFIFO "/tmp/myfifo" 
10 #define BUFES PIPE_BUF
11 
12 int main()
13 {
14     int fd,n;
15     char buff[BUFES];
16     //判斷有名管道是否已存在
17     if(access(MYFIFO,F_OK)==-1)
18     {    //若不存在,則創建可讀可寫的有名管道
19         if((mkfifo(MYFIFO,0666)<0)&&(errno != EEXIST))
20         {
21             printf("Could't create fifo\n");
22             exit(1);
23         }
24     }
25     //以只讀阻塞方式打開有名管道
26     fd = open(MYFIFO,O_RDONLY);
27     if(fd==-1)
28     {
29         printf("Open fifo error\n");
30         exit(1);
31     }
32     //不停讀取管道中的數據,如果沒用數據可讀,則一直處於阻塞狀態
33     while(1)
34     {
35         memset(buff,0,sizeof(buff));
36         if((n = read(fd,buff,BUFES))>0)
37         {
38             printf("Read '%s' from FIFO\n",buff);
39         }
40     }
41     close(fd);
42     exit(0);
43 }

  3.3 運行結果及分析

    為了能夠較好地觀察運行結果,將兩程式分別編譯後,把這兩個程式分別在兩個終端里運行,在這裡首先啟動讀管道程式。讀管道進程在建立管道後就開始迴圈地從管道里讀出內容,如果沒有數據可讀,則一直阻塞到寫管道進程向管道寫入數據。在啟動了寫管道程式後,讀進程能夠從管道里讀出用戶的輸入內容,程式運行結果如下:

     分析:

    ①對於讀進程:

    · 若該管道是阻塞打開,且當前FIFO內無數據,則對讀進程而言一直阻塞到有數據寫入。

    · 若該管道是非阻塞打開,則不論FIFO內是否有數據,都進程都會立即執行讀操作。也就是說若FIFO內無數據,那麼讀程式將立馬返回0。

    此處使用的是阻塞方式。

    ②對於寫進程:

    · 若該管道是阻塞打開,則寫進程將一直阻塞到有數據寫入。

    · 若該管道是非阻塞打開而不能寫入全部數據,則對讀進程而言只能進行部分寫入或者調用失敗。

      ③管道模式:

    · O_RDONLY:讀管道。

    · O_WRONLY:寫管道。

    · O_RDWR:讀寫管道。

    · O_NONBLOCK:非阻塞。

    · O_CREAT:如果該文件不存在,就創建一個新的文件,並使用第3個參數為其設置許可權。

    · O_EXCL:測試文件是否存在。如果使用O_CREAT|O_CREAT時文件存在,那麼將返回出錯:errno == EEXIST。

  3.4 創建兩個有名管道實現聊天程式

  伺服器端程式:

 1 /*Server.c*/
 2 #include<unistd.h>    
 3 #include<string.h>    
 4 #include<errno.h>   
 5 #include<fcntl.h>    
 6 #include<stdio.h>    
 7 #include<stdlib.h>    
 8 #include<limits.h>    
 9 
10 #define WRITE_FIFO "/tmp/readfifo" 
11 #define READ_FIFO "/tmp/writefifo" 
12 #define BUFES PIPE_BUF
13 
14 int main(int argc,char * argv[])
15 {
16     int wfd,rfd,n;
17     char buff[BUFES],readbuff[BUFES];
18     //創建管道WRITE_FIFO
19     mkfifo(WRITE_FIFO, S_IFIFO|0666);
20     //以只讀阻塞方式打開FIFO管道
21     rfd = open(READ_FIFO,O_RDONLY);
22     //以只寫阻塞方式打開FIFO管道
23     wfd = open(WRITE_FIFO,O_WRONLY);
24     //不停讀取管道中的數據,如果沒用數據可讀,則一直處於阻塞狀態
25     while(1){
26         memset(readbuff,0,sizeof(readbuff));
27         if((n = read(rfd,readbuff,BUFES))>0)
28         {
29             readbuff[BUFES] = '\0';
30             printf("Client: %s\n",readbuff);
31         }
32         memset(buff,0,sizeof(buff));
33         printf("Server: ");
34         fgets(buff, BUFES, stdin);
35         buff[strlen(buff)-1] = '\0';
36 
37         write(wfd,buff,strlen(buff));
38     }
39     close(wfd);
40     close(rfd);
41     exit(0);
42 }

  客戶端程式:

 1 /*Client.c*/
 2 #include<unistd.h>    
 3 #include<string.h>    
 4 #include<errno.h>   
 5 #include<fcntl.h>    
 6 #include<stdio.h>    
 7 #include<stdlib.h>    
 8 #include<limits.h>    
 9 
10 #define WRITE_FIFO "/tmp/writefifo" 
11 #define READ_FIFO "/tmp/readfifo" 
12 #define BUFES PIPE_BUF
13 
14 int main(int argc,char * argv[])
15 {
16     int wfd,rfd,n;
17     char buff[BUFES],readbuff[BUFES];
18     //創建管道READ_FIFO
19     mkfifo(WRITE_FIFO, S_IFIFO|0666);    
20     //以只寫阻塞方式打開FIFO管道
21     wfd = open(WRITE_FIFO,O_WRONLY);
22     //以只讀阻塞方式打開FIFO管道
23     rfd = open(READ_FIFO,O_RDONLY);
24     //不停讀取管道中的數據,如果沒用數據可讀,則一直處於阻塞狀態
25     while(1)
26     {    
27         memset(buff,0,sizeof(buff));
28         printf("Client: ");
29         fgets(buff, BUFES, stdin);
30         buff[strlen(buff)-1] = '\0';
31 
32         write(wfd,buff,strlen(buff));
33         memset(readbuff,0,sizeof(readbuff));
34         if((n = read(rfd,readbuff,BUFES))>0)
35         {
36             printf("Server: %s\n",readbuff);
37         }
38     }
39     close(wfd);
40     close(rfd);
41     exit(0);
42 }

  3.5 運行效果及分析

   分析:在這裡首先啟動客戶端程式,再啟動伺服器端程式。客戶端在建立管道後,首先寫入數據到管道中,接著數據傳遞到伺服器端,然後伺服器端將數據寫入管道中,再傳遞到客戶端,如此往複迴圈。如果沒有數據可讀,則一直阻塞到寫管道進程向管道寫入數據。

4 References


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

-Advertisement-
Play Games
更多相關文章
  • 在平時開發中經常會看到有些朋友或者同事在寫代碼時會充斥著各種for,foreach,這種程式代碼太多的話閱讀性特別差,而且還顯得特別累贅,其實在FCL中有很多幫助我們提高閱讀感的方法,而現實中很多人不會用或者說不知道,這篇我就跟大家聊一聊。 一:SelectMany 這個方法絕對是提高開發速度的一大 ...
  • 前言 我使用的是oracle 11版本的資料庫,但我使用EF Core 2.1,在使用linq進行分頁會生成Oracle 12語法的SQL,在Oracle 11下會運行報錯。 在dbcontext下OnConfiguring指定使用Oracle 11版本 預設生成12版本的sql 指定版本為11的S ...
  • 最新在搞文件的線上預覽,網上很多免費的方案都需要是電腦安裝office的,這要就很麻煩;收費的插件又太貴了。 不過還是找到一款相對好用的免費線上預覽插件。 直接在nuget上搜索ce.office.extension引入就OK了 使用十分簡單 excel 轉 html ce.office.exten ...
  • 一、引言 linux 內核的等待隊列和進程調度息息相關,進程在某些情況下必須等待某些事件的發生,例如:等待一個磁碟操作的終止,等待釋放系統資源,或等待指定的時間間隔。 等待隊列實現了在事件上的條件等待:希望等待特定事件的進程把自己放進合適的等待隊列,並放棄控制權。 因此,等待隊列表示一組睡眠的進程, ...
  • 其實在Linux伺服器中通常是不允許關機的,最多也只是定期重啟一下系統,登出也是應該養成一個離開就退出登錄的好習慣。 系統運行級別可通過/etc/inittab文件查看不同級別的含義,以及預設的級別設置。預設的級別不能設置為0(關機)和6(重啟),因為一開機就馬上關機或重啟肯定是不行的。runlev ...
  • 本文記一些常用的網路命令,包括write、ping、ifconfig、last、traceroute、netstat等命令。 write命令write 用戶名:給指定線上用戶發送信息,回車後就可以編輯需要發送的信息,編輯信息完信息後以Ctrl+D或Ctrl+c保存結束併發送(最好是在新的一行保存發送 ...
  • 一、部署MySQL 1. 搜索mysql鏡像 2. 拉取mysql鏡像 3. 創建容器,設置埠映射、目錄映射 參數說明: $PWD:表示當前目錄所在路徑,現在表示 。 p 3306:3306 :將容器的 3306 埠映射到宿主機的 3306 埠。 v $PWD/conf:/etc/mysql/ ...
  • 一、檢查電腦CPU虛擬化grep '{vmx|svm}' /proc/cpuinfo二、安裝KVM相關軟體包[root@hostname ~]#yum install -y libvirt-client #libvirt客戶端,最主要的的功能之一就是在宿主機關機時通知虛擬機也關機。[root@ho... ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...