進程間通訊——管道

進程間通訊——管道
管道簡介
常說的管道都是匿名半雙工管道,匿名半雙工管道在系統中是沒有實名的,並不能夠在文件系統中以任何方式看到該管道。它只是進程的一種資源,會隨着進程的結束而被系統清除。管道通訊是在UNIX系統中應用比較頻繁的一種方式,例如使用grep查找:ls |grep ipc
顧名思義,匿名半雙工管道沒有名字,這個特性致使了它在應用方面的侷限性:
       1.只能在有具備公共祖先的進程間通訊,即或是父子關係進程間、或是在兄弟關係進程間通訊。
       2.數據只能在一個方向上移動
管道建立
儘管這樣,半雙工管道仍是Linux系統上最多見的通訊方式。Linux系統用pipe函數建立一個半雙工管道,期函數原型爲:
        include<unistd.h>
        int pipe(int fd[2]);
     參數fd是一個長度爲2的文件描述符,fd[1]爲輸入文件描述符,fd[1]爲輸出文件描述符;函數返回類型爲int,0表示成功,-1表示失敗。當成功返回時,則自動維護一個從fd[1]到fd[0]的同行管道。
管道操做
對管道進行讀寫,能夠經過write和read進行:
            write(fd[1], "create the pipe successfully !\n", 31 ); /*向管道寫入端寫入數據*/
                     char str[256];
                     read(fd[0], str, sizeof(str) ); /*從管道讀出端讀出數據*/
       讀寫時,須要注意的問題:
                     1.當輸出端關閉時,對輸入端的寫操做會產生信號SIGPIPE,說明管道讀端已經關閉,而且write操做返回爲–1,errno的值爲EPIPE,對於SIGPIPE信號能夠進行捕捉處理。
            2.read返回0說明管道中沒數據,但不能說明輸入端是否關閉
操做流程
    1.建立管道
    2.利用fork建立子進程
    3.控制管道流向
    4.從fd[1]向管道寫信息
    5.經過fd[0]從管道渡信息
建立管道的標準庫函數
     #include <stdio.h>
         FILE *popen( const char * command, const char *mode ); 
         int pclose ( FILE *stream );  
示例
引用《Linux C程序設計大全》 中的例子

  1. #include <unistd.h>
  2. #include <stdio.h>
  3. #include <fcntl.h>
  4. #include <sys/types.h>
  5. #define BUFES PIPE_BUF /* PIPE_BUF管道默認一次性讀寫的數據長度*/
  6. int main ( void ) {
  7.          int fd[2];
  8.          char buf[BUFSZ];
  9.          pid_t pid;
  10.          int len;
  11.          if ( (pipe(fd)) < 0 ){ /*建立管道*/
  12.                   perror ( "failed to pipe" );
  13.                   exit( 1 );
  14.          }
  15.          if ( (pid = fork()) < 0 ){ /* 建立一個子進程 */
  16.                   perror ( "failed to fork " );
  17.                   exit( 1 );
  18.          }
  19.          else if ( pid > 0 ){
  20.                 close ( fd[0] ); /*父進程中關閉管道的讀出端*/
  21.                  write (fd[1], "hello my son!\n", 14 ); /*父進程向管道寫入數據*/
  22.                  exit ( 0);
  23.          }
  24.          else {
  25.                  close ( fd[1] ); /*子進程關閉管道的寫入端*/
  26.                 len = read (fd[0], buf, BUFS ); /*子進程從管道中讀出數據*/
  27.         if ( len < 0 ){
  28.                  perror ( "process failed when read a pipe " );
  29.                  exit( 1 );
  30.         }
  31.          else
  32.                  write(STDOUT_FILENO, buf, len); /*輸出到標準輸出*/
  33.                  exit(0);
  34.          }
  35. }
相關文章
相關標籤/搜索