UNIX高级编程--管道

管道

管道是 UNIX 系统 IPC 的最高老形式,所有的 UNIX 系统都提供此种通信机制。管道有以下两种局限性。

  1. 历史上,它们是半双工(既数据只能在一个方向上流动)。现在,某些系统提供全双工管道,但是为了最佳的可移植性,我们决不应预先假定系统支持全双工管道。
  2. 管道只能在具有公共祖先的两个进程之间使用。通常,一个管道由一个进程创建,在进程调用 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;
}

在上面的例子中,直接对管道描述符调用 readwrite 。更有趣的是将管道描述符复制到标准输入或标准输出上。通常,子进程会在此之后执行另一个程序,该程序或者从标准输入(已创建的管道)读数据,或者将数据写至其标准输出(该管道)。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值