管道
管道是 UNIX 系统 IPC 的最高老形式,所有的 UNIX 系统都提供此种通信机制。管道有以下两种局限性。
- 历史上,它们是半双工(既数据只能在一个方向上流动)。现在,某些系统提供全双工管道,但是为了最佳的可移植性,我们决不应预先假定系统支持全双工管道。
- 管道只能在具有公共祖先的两个进程之间使用。通常,一个管道由一个进程创建,在进程调用
fork
之后,这个管道就能在父进程之间使用了。
FIFO
没有第二种局限性,UNIX 域套接字没有这两种局限性。
尽管有这两种局限性,半双工管道仍是最常用的 IPC 形式。每当在管道中键入一个命令序列,让 shell 执行时,shell 都会为每一条命令单独创建一个进程,然后用管道将前一条命令进程的标准输出与后一条命令的标准输入相连接。
函数
#include <unistd.h>
int pipe(int pipefd[2]);
// 返回值:成功,返回 0,出错,返回 -1
- fd 返回两个文件描述符:
fd[0]
为读而打开,fd[1]
为写而打开。fd[1]
的输出是fd[0]
的输入。
单个进程中的管道几乎没有任何用处。通常,进程会先调用 pipe
,接着调用 fork
,从而创建从父进程到子进程的 IPC 通道,反之亦然。
实现一个 shell 命令的调用,子进程调用 shell 命令,将输出利用管道 传递给父进程输出。
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <fcntl.h>
#include <string.h>
#include <sys/types.h>
int main(int arg,char *argv[]){
int fd[2];
if(pipe(fd)<0){
printf("pipe error...\n");
return -1;
}
pid_t pid;
if((pid=fork())<0){
printf("fork error...\n");
return -1;
}
if(pid==0){
char *agv[arg];
for(int i=1;i<arg;++i) agv[i-1]=argv[i];
agv[arg-1]=NULL;
close(fd[0]);
dup2(fd[1],STDOUT_FILENO);
if(execvp(agv[0],agv)<0){
printf("exec error...\n");
}
return 0;
}else{
close(fd[1]);
char buf[4096];
int l=read(fd[0],buf,4096);
printf("%s\n",buf);
return 0;
}
return 0;
}
在上面的例子中,直接对管道描述符调用 read
和 write
。更有趣的是将管道描述符复制到标准输入或标准输出上。通常,子进程会在此之后执行另一个程序,该程序或者从标准输入(已创建的管道)读数据,或者将数据写至其标准输出(该管道)。