成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

Linux進程通信之管道解析

系統 Linux
所謂的管道,也就是內核里面的一串緩存,從管道的一段寫入的數據,實際上是緩存在內核中的,令一端讀取,也就是從內核中讀取這段數據。

 [[409600]]

概述

管道是 UNIX系統 IPC的最古老的形式,所有的UNIX系統都提供此種通信。所謂的管道,也就是內核里面的一串緩存,從管道的一段寫入的數據,實際上是緩存在內核中的,令一端讀取,也就是從內核中讀取這段數據。對于管道傳輸的數據是無格式的流且大小受限。對于管道來說,也分為匿名管道和命名管道,其中命名管道也被叫做 FIFO,下面則分別闡述這兩種管道。

匿名管道

默認情況下,在 Shell命令執行過程中,任何一個命令都有一個標準輸入設備(鍵盤)、標準輸出設備(顯示器)和標準輸出設備(顯示器),使用管道"|"可以將兩個命令連接起來,從而改變標準的輸入輸出方式,下面是在 Linux 端運行命令行的一個截圖:

上述命令中的意思也就是,將ls命令得到的結果作為 grep tags命令的輸入。

連接輸入輸出的中間設備即為一個管道文件,綜上,也就是說使用管道可以將一個命令的輸出作為另一個命令的輸入(在運行的時候,一個命令將創建一個進程),而這種管道是臨時的,命令執行完畢之后就會自動消失,這類管道稱為無名管道。

匿名管道例子

匿名管道在使用前要先創建,其函數的聲明如下:

  1. extern int pipe (int __pipedes[2]); 

此函數的參數是一個整型數組,如果執行成功,pipe 將存儲兩個整型文件描述符于__pipedes[0]和__pipedes[1]中,他們分別指向管道的兩端。如果系統調用失敗,則返回 -1。

讀無名管道,該函數的聲明如下:

  1. extern ssize_t read (int __fd, void *__buf, size_t __nbytes); 

第一個參數fd為打開的文件描述符,buf為讀出數據的存儲位置,nbytes為讀取數據的大小,調用 read 函數將從 fd 指向的文件描述符指定的打開文件中宏讀 n 字節到 buf 指向的緩沖區內。

如果試圖向已經填滿的管道寫入,系統會自動阻塞。一個管道不能同時被兩個進程打開。

  1. extern ssize_ t write(int __fd, __const void *__buf, size_t __n); 

從 buf指向的緩沖區中向管道中寫入nbytes字節,且每次寫入的內容都附件在管道的末端。

那要如何使用管道在兩個進程之間通信呢,我們可以使用 fork()創建子進程,創建的子進程會復制父進程的文件描述符,這樣就做到了兩個進程各有兩個fd[0]與fd[1],兩個進程就可以通過各自的fd寫入和讀取同一個管道文件實現進程通信了,具體原理如下所示:

具體的例子如下所示:

  1. #include<unistd.h> 
  2. #include<stdio.h> 
  3. #include<stdlib.h> 
  4.  
  5. int main(int argc, char *argv[]) 
  6.     pid_t pid; 
  7.     int temp
  8.     int pipedes[2]; 
  9.     char s[14] = "test message!"
  10.     char d[14]; 
  11.  
  12.     if (pipe(pipedes) == -1) // 創建管道 
  13.     { 
  14.         perror("pipe"); 
  15.         exit(EXIT_FAILURE); 
  16.     } 
  17.  
  18.     if (pid == fork() == -1) 
  19.     { 
  20.         perror("fork"); 
  21.         exit(EXIT_FAILURE); 
  22.     } 
  23.     else if (pid == 0)      // 子進程 
  24.     { 
  25.         printf("now,write data to pipe\n"); 
  26.         if (write(pipedes[1], s, 14) == -1)   // 寫數據到管道 
  27.         { 
  28.             perror("write"); 
  29.             exit(EXIT_FAILURE); 
  30.         } 
  31.         else 
  32.         { 
  33.             printf("the written data is:%s\n",s); 
  34.             exit(EXIT_SUCESS); 
  35.         } 
  36.     } 
  37.     else if (pid > 0)     // 父進程 
  38.     { 
  39.         slepp(2); 
  40.         printf("now, read from pipe\n"); 
  41.         if ((read(pipedes[0], d, 14)) == -1) 
  42.         { 
  43.             perror("read"); 
  44.             exit(EXIT_FAILURE); 
  45.         } 
  46.         printf("the data from pipe is:%s\n",d); 
  47.     } 
  48.     return 0; 

代碼運行的結果如下所示:

命名管道

命名管道又被稱之為是 FIFO ,未命名的管道只能在兩個相關的進程之間使用,而且這兩個相關的進程還要又一個共同創建了他們的祖先進程,但是,通過 FIFO ,不相關的進程也能交換數據。

首先,介紹下是如何創建命名管道的:

  1. extern int mkfifo (__const char *__path, __mode_t __mode); 

mkfifo會根據參數建立特殊的有名管道文件,該文件必須不存在,而參數mode為該文件的權限。

下面是一個使用命名管道進行進程間通信的例子,例子分為兩個程序,分別是讀部分和寫部分,首先看先往管道寫數據的代碼,代碼如下所示:

  1. #include <stdio.h>  
  2. #include <string.h>  
  3. #include <fcntl.h>  
  4. #include <sys/stat.h>  
  5. #include <sys/types.h>  
  6. #include <unistd.h>  
  7.  
  8. int main()  
  9. {  
  10.     int fd;  
  11.  
  12.     // FIFO file path  
  13.     char * myfifo = "/tmp/myfifo";  
  14.  
  15.     // Creating the named file(FIFO)  
  16.     // mkfifo(<pathname>, <permission>)  
  17.     mkfifo(myfifo, 0666);  
  18.  
  19.     char arr1[80], arr2[80];  
  20.     while (1)  
  21.     {  
  22.         // Open FIFO for write only  
  23.         fd = open(myfifo, O_WRONLY);  
  24.         printf("The fd is:%d\n",fd); 
  25.  
  26.         // Take an input arr2ing from user.  
  27.         // 80 is maximum length  
  28.         fgets(arr2, 80, stdin);  
  29.  
  30.         // Write the input arr2ing on FIFO  
  31.         // and close it  
  32.         write(fd, arr2, strlen(arr2)+1);  
  33.         close(fd);  
  34.  
  35.         // Open FIFO for Read only  
  36.         fd = open(myfifo, O_RDONLY);  
  37.  
  38.         // Read from FIFO  
  39.         read(fd, arr1, sizeof(arr1));  
  40.  
  41.         // Print the read message  
  42.         printf("User2: %s", arr1);  
  43.         close(fd);  
  44.     }  
  45.     return 0;  

然后是先往管道讀數據的代碼,代碼如下所示:

  1. #include <stdio.h>  
  2. #include <string.h>  
  3. #include <fcntl.h>  
  4. #include <sys/stat.h>  
  5. #include <sys/types.h>  
  6. #include <unistd.h>  
  7.  
  8. int main()  
  9. {  
  10.     int fd1;  
  11.  
  12.     // FIFO file path  
  13.     char * myfifo = "/tmp/myfifo";  
  14.  
  15.     char str1[80], str2[80];  
  16.     while (1)  
  17.     {  
  18.         // First open in read only and read  
  19.         fd1 = open(myfifo,O_RDONLY);  
  20.         printf("The fd is:%d\n",fd1); 
  21.         read(fd1, str1, 80);  
  22.  
  23.         // Print the read string and close  
  24.         printf("User1: %s", str1);  
  25.         close(fd1);  
  26.  
  27.         // Now open in write mode and write  
  28.         // string taken from user.  
  29.         fd1 = open(myfifo,O_WRONLY);  
  30.         fgets(str2, 80, stdin);  
  31.         write(fd1, str2, strlen(str2)+1);  
  32.         close(fd1);  
  33.     }  
  34.     return 0;  
  35. }  

下面是代碼運行的一個結果:

說明一下,就是說當運行 write程序的時候,會創建fifo文件,命名管道,然后,在 write文件中就執行open操作,但是,這里存在的一個問題就是,因為在運行 write程序的時候,沒有進程打開讀端,也就阻塞了 open函數的運行,只有運行read操作,以讀的方式讀取管道的數據,這樣才能使得write中的open函數繼續執行。

綜上,也就是命名管道在進程中通信的一個例子。

小結

上述就是本次進程通信中關于管道的相關內容,其中就包括匿名管道以及命名管道,他們之間存在著差別嗎,也各有各的應用,本次的分享就到這里啦~

本文轉載自微信公眾號「wenzi嵌入式軟件」,可以通過以下二維碼關注。轉載本文請聯系wenzi嵌入式軟件公眾號。

 

責任編輯:武曉燕 來源: wenzi嵌入式軟件
相關推薦

2021-09-30 10:45:33

Linux進程通信

2019-05-13 10:00:41

Linux進程間通信命令

2010-03-02 10:27:56

Linux進程狀態

2010-03-02 10:43:45

Linux進程狀態

2010-01-05 10:00:48

Linux進程間通信

2011-06-22 16:50:09

Qt 進程 通信機制

2010-01-21 11:22:35

Linux多線程同步

2023-03-05 16:12:41

Linux進程線程

2011-06-24 14:01:34

Qt QCOP 協議

2018-01-12 14:35:00

Linux進程共享內存

2017-06-19 13:36:12

Linux進程消息隊列

2009-12-24 14:47:42

Linux系統進程

2011-06-22 17:27:19

QT 進程通信

2023-03-02 23:50:36

Linux進程管理

2021-02-14 21:05:05

通信消息系統

2009-02-23 15:55:29

ASP.NET.NET性能提升

2011-01-11 13:47:27

Linux管理進程

2011-08-08 10:02:55

iPhone開發 進程 通信

2023-03-03 00:03:07

Linux進程管理

2018-05-30 13:58:02

Linux進程通信
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 秋霞精品 | 国产精品久久久久久亚洲调教 | 黄色在线免费观看 | 一区二区中文 | 一二三区在线 | 综合久久综合久久 | 欧美激情亚洲激情 | 国产欧美一区二区三区久久 | 一区欧美 | 三级视频在线观看电影 | 欧美一区二区在线观看 | 久草在线 | 色婷婷一区二区三区四区 | 久久亚洲国产 | 日韩精品一区二区三区四区视频 | 一久久久| 久久久久久久久久一区二区 | 本地毛片| 久久久久久久久久久久91 | 色婷婷激情综合 | 免费九九视频 | 国产区在线看 | 红桃视频一区二区三区免费 | 日韩视频专区 | 成人免费网站www网站高清 | 国产精品亚洲成在人线 | 麻豆精品国产91久久久久久 | 毛片免费在线 | 麻豆精品久久 | 国产高清精品一区二区三区 | 国产精品无码久久久久 | 国产欧美精品一区二区 | 欧美电影免费观看高清 | 国产精品美女久久久久久免费 | 欧美一区二区三区国产 | 国产一区二区视频在线观看 | 一区二区三区国产 | av在线免费观看网站 | a黄视频 | 一区二区三区四区在线视频 | 91夜色在线观看 |