linux进程间通信-管道

一 管道的局限性

    管道有两个局限性:(1)他是半双工(即数据只能在一个方向上流动)。(2)它只能在具有公共祖先的进程之间使用。一个管道由一个进程创建,然后该 进程调用fork,此后父子进程之间就可该管道。

 

二 管道的创建

   用函数pipe创建:
#include<unistd.h>
int pipe(int files[2]);
参数:filedes返回两个文件描述符:filedes[0] 为从管道读而打开,filedes[1]为管道写而打开。filedes[1]的输出是filedes[0]的输入。

三 管道原理图

  如下这张图可以说明管道的原理,即一个父进程创建一个子进程后,父进程打开写管道,子进程打开读管道。

四 管道只有一端时的情况

当管道的一端被关闭后,下列规则起作用:
(1)  当读一个写端已被关闭的管道时,在所有数据都被读取后, read返回0,以指示达到了文件结束处(从技术方面考虑,管道的写端还有进程时,就不会产生文件的结束。可以复制一个管道的描述符,使得有多个进程具有写打开文件描述符。但是,通常一个管道只有一个读进程,一个写进程。下一节介绍F I F O时,我们会看到对于一个单一的FIFO常常有多个写进程)。
(2)  如果写一个读端已被关闭的管道,则产生信号 SIGPIPE。如果忽略该信号或者捕捉该信号并从其处理程序返回,则 write 出错返回,errno设置为EPIPE。
(3) 在写管道时,常数 PIPE_BUF 规定了内核中管道缓存器的大小。如果对管道进行 write调用,而且要求写的字节数小于等于 PIPE_BUF,则此操作不会与其他进程对同一管道(或 FIFO )的write操作穿插进行。但是,若有多个进程同时写一个管道(或 FIFO) ,而且某个或某些进程要求写的字节数超过 PIPE_BUF字节数,则数据可能会与其他写操作的数据相穿插。
 

五 例子

#include <unistd.h>
#include <stdio.h>
#include <stdlib.h>
 
#define MAXLINE (2014)
 
int main(void)
{
    int n, fd[2];
    pid_t pid;
    char line[MAXLINE];
 
    if (pipe(fd) < 0)
        printf("pipe error");
 
    if((pid = fork()) < 0)
        printf("fork error");
     else if (pid > 0) 
     { /* parent */
        close(fd[0]);/* close read */
        printf ("the process pid %d\n", getpid());
        printf ("the process write to pipe : hello world!\n");
        write(fd[1], "hello world\n", 12);
     } 
    else
    { /* child */
         close(fd[1]); /* close write */
         printf("the process pid %d\n", getpid());
         n = read(fd[0], line, MAXLINE);
         write(STDOUT_FILENO, line, n);
    }
 
    exit(0);
}
 

 

 
 
 
 
 
 
 
 
posted @ 2014-10-21 11:06  xyecho  阅读(650)  评论(0编辑  收藏  举报