Linux进程间的通信

1.进程间通信介绍

1.1 进程间通信目的:一个进程需要将他的数据发送给另一个进程,大家应该都多少接触过linux中的管道符"|",这个符号就是用来多个命令执行,在Linux中每一个命令都是一个独立的进程,这里我们可以在Linux中看到这些命令,都是一个个的可执行程序,每次输入时相当于一个可执行程序开始跑。
在这里插入图片描述
我们这里切换到ls这个命令的目录下,在ubuntu系统下我们看到了熟悉的apt安装命令,还有很多比较冷门的命令,我们学习阶段接触的比较少,但是我们可以发现这都是一些可执行程序。
这里我们还可以使用ldd命令查一下,这些命令都是用什么写的:
在这里插入图片描述
使用ldd命令可以查看这个可执行程序使用的依赖哪些库产生,这里明显的看出是依赖的c语言的动态库编写的。
所以这里我们需要理解,每一个命令都是一个可执行程序。
在这里插入图片描述

2.匿名命名管道原理操作

讲到原理,那么为什么可以使用管道符"|"一起执行两个可执行程序呢,这里就是通信的力量啦。
那么什么是通信呢?
在这里插入图片描述
这里两个进程就可以指向同一块空间,这样我们就具备了最基本的沟通能力。
有了沟通的能力我们就可以想办法让他们可以传递信息。
这里我们要介绍一个接口pipe。

//此处以C++举例
#include<unistd.h>
int pipe(int fd[2]);

pipe函数定义中的fd参数是一个大小为2的一个数组类型的指针。该函数成功时返回0,并将一对打开的文件描述符值填入fd参数指向的数组。失败时返回 -1并设置errno。
在这里插入图片描述
下面是一段验证原理的代码,使用c++编写,最后实现了管道的单向通信:

#include <iostream>
#include <unistd.h>
#include <sys/types.h>
#include <sys/wait.h>
#include <cstdio>
#include <cstring>
using namespace std;

void write_pipe(int wfd)
{
    char buffer[1024];
    int cnt = 0;
    while (true)
    {
        snprintf(buffer, sizeof(buffer), "I am god! 我的pid:%d, cnt=%d", getpid(), cnt++);
        ssize_t ret = write(wfd, buffer, strlen(buffer));
        if (ret < 0)
        {
            perror("write");
            break;
        }
        sleep(1);
    }
}

void read_pipe(int rfd)
{
    char buffer[1024];
    while (true)
    {
        ssize_t n = read(rfd, buffer, sizeof(buffer) - 1);
        if (n > 0)
        {
            buffer[n] = '\0';
            cout << "child say: " << buffer << endl;
        }
        else if (n == 0)
        {
            cout << "pipe closed" << endl;
            break;
        }
        else
        {
            perror("read");
            break;
        }
    }
}

int main()
{
    int fds[2] = {0};
    if (pipe(fds) < 0)
    {
        perror("pipe");
        return 1;
    }

    pid_t id = fork();
    if (id < 0)
    {
        perror("fork");
        return 1;
    }

    if (id == 0)
    {                  // 子进程
        close(fds[0]); // 关闭读端
        write_pipe(fds[1]);
        close(fds[1]);
        exit(0);
    }
    else
    {                  // 父进程
        close(fds[1]); // 关闭写端
        read_pipe(fds[0]);
        close(fds[0]);
        waitpid(id, nullptr, 0);
    }

    return 0;
}

在这里插入图片描述
这里我们写了一个shell脚本用来观察我们写的程序,发现是可以进行读写。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值