进程间通信(一):使用普通管道pipe实现父子、兄弟进程间的通信

一、pipe管道的特点

1)内核借助环形队列机制,使用内核缓冲区实现
2)管道中的数据只能读取一次,不可反复读
3)半双工通信,数据在管道中,只能单向流动,即一端进行读操作时另一端只能进行写操作,反之,一端进行写操作是另一端只能进行读操作
4)只能用于具有血缘关系的进程,如父子进程、兄弟进程等

二、pipe函数

功能:创建并打开管道

int pipe(int fd[2])

其中fd[0]表示读端,fd[1]表示写端,成功返回0,失败返回-1。

三、管道的读写行为

1.读端

1)若管道有数据,read返回实际读到的字节数;
2)若管道无数据
I. 无写端,read返回0(类似读到文件尾)
II. 有写端,read阻塞等待

2.写端

1)无读端,则异常终止;
2)有读端
I. 管道已满,阻塞等待
II. 管道未满,返回写出字节数

四、父子进程通信实现案例

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <sys/types.h>
#include <unistd.h>
#include <pthread.h>

void sys_err(const char *str)
{
    perror(str);
    exit(1);
}

int main(void)
{
    int ret;
    int fd[2];
    pid_t pid;
    char *str = "hello pipe\n";
    char buf[1024];
    ret = pipe(fd);
    if(ret == -1) sys_err("pipe error");
    
    pid = fork();
    if(pid > 0)//父进程
    {
        close(fd[0]);//close read
        write(fd[1], str, strlen(str));//将str的内容写入管道
        sleep(1);
        close(fd[1]);
    }
    else//子进程
    {
        close(fd[1]);//close write
        ret = read(fd[0], buf, sizeof(buf));//从管道中读取并存入buf
        write(STDOUT_FILENO, buf, ret);//将读取存入buf的内容写入标准输出
        close(fd[0]);
    }
    return 0;
}

程序执行后会在终端显示如下内容

hello pipe

五、兄弟进程通信实现案例

该案例使用管道实现如下命令:

vim --version | grep python

代码如下:

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <sys/types.h>
#include <unistd.h>
#include <pthread.h>
#include <errno.h>
#include <sys/wait.h>

void sys_err(const char *str)
{
    perror(str);
    exit(1);
}

int main(int argc, char *argv[])
{
    int i;
    int fd[2];
    int ret;
    pid_t pid;

    ret = pipe(fd);
    if(ret == -1)
    {
        sys_err("pipe error");
    }
    for(i=0; i<2; i++)
    {
        pid = fork();
        if(pid == -1)
        {
            sys_err("fork error");
        }
        if(pid == 0)
        {
            break;
        }
    }
    if(i == 2)//父
    {
    	//为保证管道中数据的单向流动,需在父进程中将读端和写端都关闭
        close(fd[0]);
        close(fd[1]);
        wait(NULL);
        wait(NULL);
    }
    else if(i == 0)//兄
    {
        close(fd[1]);//关闭写端
        //将标准输入内容重定向至管道读端,wc命令默认从标准输入中读取内容,
        //则重定向后wc命令会从管道中读取内容
        dup2(fd[0], STDIN_FILENO);
        execlp("grep", "grep", "python", NULL);
        sys_err("execlp grep error");
    }
    else if(i == 1)//弟
    {
        close(fd[0]);//关闭读端
        //将标准输出内容重定向至管道写端,则vim --version得到的结果会写入管道
        dup2(fd[1], STDOUT_FILENO);
        execlp("vim", "vim", "--version", NULL);
        sys_err("execlp ls error");
    }

    return 0;
}

程序执行结果:

+comments          +libcall           -python            +vreplace
+conceal           +linebreak         +python3           +wildignore
链接方式: gcc   -Wl,-E  -Wl,-Bsymbolic-functions -Wl,-z,relro -Wl,-z,now -Wl,--as-needed -o vim   -lgtk-3 -lgdk-3 -lpangocairo-1.0 -lpango-1.0 -latk-1.0 -lcairo-gobject -lcairo -lgdk_pixbuf-2.0 -lgio-2.0 -lgobject-2.0 -lglib-2.0 -lSM -lICE -lXpm -lXt -lX11 -lXdmcp -lSM -lICE  -lm -ltinfo -lnsl  -lselinux  -lacl -lattr -lgpm -ldl  -L/usr/lib -llua5.2 -Wl,-E  -fstack-protector-strong -L/usr/local/lib  -L/usr/lib/x86_64-linux-gnu/perl/5.26/CORE -lperl -ldl -lm -lpthread -lcrypt  -L/usr/lib/python3.6/config-3.6m-x86_64-linux-gnu -lpython3.6m -lpthread -ldl -lutil -lm -L/usr/lib/x86_64-linux-gnu -ltcl8.6 -ldl -lz -lpthread -lm     
  • 3
    点赞
  • 6
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 无名管道(Anonymous Pipe)是操作系统中用于实现进程间通信(IPC)的一种机制,Windows操作系统也支持使用无名管道进行进程间通信。 在Windows操作系统中,无名管道是一种特殊类型的文件,只能用于相关进程间通信,无法用于不相关进程间通信。无名管道通常是单向的,有一个读取端(Read end)和一个写入端(Write end),一个进程可以往管道的写入端写入数据,另一个进程可以从管道的读取端读取数据。 无名管道常用于兄弟进程间通信(如父子进程)或者由同一个进程创建的两个进程间通信使用无名管道进行进程间通信的步骤如下: 1. 创建管道:调用CreatePipe函数创建一个无名管道,返回读取端和写入端句柄。 2. 创建进程使用CreateProcess函数创建需要进行通信进程。 3. 设置进程的标准输入输出:使用SetStdHandle函数将管道的读取端和写入端句柄分别设置为所创建进程的标准输入和标准输出。 4. 进程通信:通过写入端句柄向管道写入数据,然后通过读取端句柄从管道读取数据。 在这个过程中,写入端和读取端可以位于同一个进程中,也可以位于不同的进程中。写入端写入的数据会被读取端读取,实现进程通信。 无名管道可以用于传输任意类型的数据,包括文本、二进制等。只要进程通过读取端和写入端的兼容接口进行数据的读写操作,就可以实现进程通信。 总之,使用无名管道是一种简单有效的方式来实现Windows操作系统中进程通信,通过读取端和写入端的数据传输,可以实现数据的共享和交换,满足进程间通信的需求。 ### 回答2: 无名管道(Anonymous Pipes)是一种在Windows系统中用于实现进程间通信的机制。它通常用于在父子进程或者兄弟进程进行数据传输。 在使用无名管道之前,我们需要创建一个匿名的管道对象,它定义了一个读取端和一个写入端。进程可以通过创建管道对象,并将写入端传递给子进程,来实现进程间通信子进程可以通过读取从进程写入端传来的数据来获取信息。 无名管道使用步骤如下: 1. 使用CreatePipe函数创建一个无名管道,该函数返回两个句柄(一个读句柄和一个写句柄),分别用于读取和写入管道中的数据。 2. 通常,我们会在进程中调用CreateProcess函数创建一个子进程子进程会继承进程的句柄。 3. 在创建子进程后,进程可以关闭子进程中不需要的句柄(写句柄),子进程可以关闭进程中不需要的句柄(读句柄)。 4. 当进程想要向子进程发送数据时,它可以使用WriteFile函数来向写入端写入数据。 5. 子进程可以使用ReadFile函数从读取端读取进程写入的数据。 通过这种方式,进程子进程可以进行双向的通信。 需要注意的是,无名管道只适用于具有亲密关系(比如父子进程)或具有共同祖先(比如兄弟进程)的进程。此外,无名管道是半双工的,意味着只能在一个方向上传递数据。如果需要实现双向通信,可以使用两个无名管道来分别实现

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值