【linux】进程间通信(命名管道 && System V)

我们在上一节知道了匿名管道的原理与它对应的情况与特征。

在这一节将会对命名管道 以及system V标准下的IPC进行一下原理的讲解,尤其是对共享内存的理解。

目录

命名管道:

先输出一个结论:

命名管道与匿名管道的情况与特征完全一致,除了匿名管道只能与有血缘关系的进程建立,而命名管道无这个需求。

原理我们来看一看:

下图是文件系统和进程管理的示意图:

如果我们的进程B也打开同一个文件。

进程 = 内核数据结构 + 代码与数据

故PCB与文件描述表每个进程有自己独立的一份是我们理解并接受的。

但是为什么struct file为什么有两份?
答:

其一:每个进程的文件描述表是独立的,故里面的指针也是不同的

其二:如果进程A以"w"方式打开,那么struct file内部一定会有一个表示flag的字段,此时进程B若是以"r"打开就肯定是独立的两份。

这个理解了我们再来探讨一下为什么struct file的指向是一样的?
答:

因为OS不会做浪费时空的事情,磁盘内文件的属性,方法,以及内核缓冲区绝大部分都是一样的,再重新搞一份是没有必要的,

这也就为我们为命名管道的理解有了切入点,也就是共同的内核缓冲区

但是命名管道与普通文件肯定是不一样的,命名管道至少不会向磁盘中写入数据,但是这个磁盘文件可以帮助我们定位,使两个进程以同一个路径找到。

我们先来看一看命令行中的命名管道。

我们先创建一个名为myfifo的命名管道,在管道内写完消息后,可以在另一端读出,echo是一个进程,cat是一个进程,这就完成了两个进程之间的通信。

另外我们也可以继续向管道内写入信息,并发现大小为0.

也可以证明这是一个特殊的文件。

那么如何在代码中进行创建呢?

这就用到了我们刚刚说到的要规定一个路径,用路径进行定位。

创建好了如何打开?使用open,如何写与读,使用write与read

我们可以看看下段这边的代码,实现了一个类,对以上的各种操作进行了封装。

cpp 复制代码
#pragma once

#include <iostream>
#include <string>
#include <cstring>
#include <cerrno>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <unistd.h>

#define CommPath "./fifopipe"
#define Read O_RDONLY
#define Write O_WRONLY
#define Creater 1
#define User 2
#define filename "fifopipe"
#define Default -1
#define Size 4096

class NamedPipe
{
private:
    bool OpenNamedPipe(int mode)
    {
        _fd = open(filename, mode);
        if (_fd > 0)
        {
            return true;
        }
        else
        {
            return false;
        }
    }

public:
    NamedPipe(std::string path, int who)
        : _path(path), _who(who), _fd(Default)
    {
        if (_who == Creater)
        {
            int res = mkfifo(_path.c_str(), 0666);
            if (res != 0)
            {
                perror("mkfifo");
            }
        }
    }
    bool OpenForRead()
    {
        return OpenNamedPipe(Read);
    }
    bool OpenForWrite()
    {
        return OpenNamedPipe(Write);
    }

    int WriteOnNamedPipe(const std::string& in)
    {
        return write(_fd, in.c_str(), Size);
    }

    int ReadOnNamedPipe(std::string* out)
    {
        char message[Size];
        int n = read(_fd, message, Size);
        *out = message;
        return n;
    }

    ~NamedPipe()
    {
        if (_who == Creater)
        {
            int res = unlink(_path.c_str());
            if (res != 0)
            {
                perror("unlink");
            }
        }
        close(_fd);
    }

private:
    std::string _path;
    int _fd;
    int _who;
};

此时我们再回归一下概念:进程间通信的前提是要先看到同一块内存空间。

匿名管道是通过继承,而命名管道是通过同一个路径。

System V:

共享内存:

消息队列(原理):

信号量(原理):

相关推荐
Web3探索者11 小时前
可视化服务器管理和传统命令行区别是什么?新手教程:Linux 运维到底该用图形界面还是 SSH 命令行?
linux·ssh
zylyehuo13 小时前
Linux系统中网线与USB网络共享冲突
linux
荣--15 小时前
一键部署不是为了省时间 —— 它是把"买来的 PaaS"变成"自己的平台"的拐点
运维·zabbix·工程化·一键部署·平台化·边界设计
江华森15 小时前
动手实战学 Docker — 从零到集群编排完全指南
运维
Avan_菜菜1 天前
FRP 内网穿透完整实战:从 HTTP 映射到 HTTPS 自签代理
运维·nginx·https
Sokach10152 天前
Linux Shell 脚本从零到能用:一个新手的一天学习总结
linux
SelectDB2 天前
Litefuse 开源并推出单进程轻量模式,25 秒就能跑起来的 Agent 可观测与评估平台
运维·后端·自动化运维
AlfredZhao2 天前
Docker 容器时区不对,`timedatectl` 不存在怎么办?
linux·timezone
zzzzzz3104 天前
9K Star 炸裂开源!这个 C 语言写的代码知识图谱,把 Linux 内核索引压缩到了 3 分钟
linux·服务器·sql
XIAOHEZIcode4 天前
Linux系统鼠标偏移常见原因以及修复方案
linux·运维·游戏