进程间通信(命名管道 共享内存)

news2024/11/5 10:59:13

文章目录

  • 命名管道
    • 原理
    • 命令创建命名管道
    • 函数创建命名管道
  • 共享内存
    • 原理
    • shmget
      • FIOK
    • 代码应用:
      • prems
      • nattch

命名管道

用于两个毫无关系的进程间的通信。

原理

Linux文件的路径是多叉树,故文件的路径是唯一的。
让内核缓冲区不用刷新到磁盘中,一旦刷新就拖慢了操作系统。所以磁盘中有个特殊文件,在内存中写入不会刷新到磁盘,让两个进程在内存中通信,该文件叫做命名管道。(命名:有路径就有名字;管道:用于内存通信)
在这里插入图片描述

命令创建命名管道

可以直接使用系统命令创建命名管道
在这里插入图片描述
p打头的文件就是管道文件
在这里插入图片描述
echo是进程,cat也是进程。两个进程基于管道实现了通信。写入的时候,管道的大小依旧为0。
在这里插入图片描述

函数创建命名管道

在这里插入图片描述
返回值:成功返回0,失败返回-1
在这里插入图片描述
删除指定路径的文件:unlink
在这里插入图片描述
返回值:成功返回0,失败返回-1
在这里插入图片描述
代码如下:
client.cc 负责写:

#include "namedPipe.hpp"

// client write
int main()
{
    NamePiped fifo(comm_path, User);
    if (fifo.OpenForWrite())
    {
        std::cout << "client open namd pipe done" << std::endl;
        while (true)
        {
            std::cout << "Please Enter > ";
            std::string message;
            getline(std::cin, message);
            fifo.WriteNamedpipe(message);
        }
    }
    return 0;
}

namePipe.hpp:

#pragma once

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

const std::string comm_path = "./myfifo";//管道的名字
#define DefaultFd -1
#define Creater 1
#define User 2
#define Read O_RDONLY
#define Write O_WRONLY
#define BaseSize 4096//一次读多少字节

class NamePiped
{
private:
    bool OpenNamedPipe(int mode)//权限
    {
        _fd = open(comm_path.c_str(),mode);
        if(_fd < 0)
            return false;
        return true;
    }
public:
    NamePiped(const std::string &path, int who) : _fifo_path(path), _id(who),_fd(DefaultFd)
    {
        if (_id == Creater)
        {
            int res = mkfifo(_fifo_path.c_str(), 0666);
            if (res != 0)
            {
                perror("mkfifo");
            }
            std::cout << "Creater creat Named pipe" << std::endl;
        }
    }
    bool OpenForRead()
    {
        return OpenNamedPipe(Read);
    }
    bool OpenForWrite()
    {
        return OpenNamedPipe(Write);
    }
    int ReadNamedpipe(std::string *out)
    {
        char buffer[BaseSize];
        int n = read(_fd,buffer,sizeof(buffer));
        if(n > 0)
        {
            buffer[n] = 0;
            *out = buffer;
        }
        return n;
    }
    int WriteNamedpipe(const std::string &in)
    {
        return write(_fd,in.c_str(),in.size());
    }
    ~NamePiped()
    {
        if (_id == Creater)
        {
            sleep(5);
            int res = unlink(_fifo_path.c_str());
            if (res != 0)
            {
                perror("unlink");
            }
            std::cout << "Creater free Named pipe" << std::endl;
        }
        if(_fd != DefaultFd) close(_fd);
    }

private:
    const std::string _fifo_path;  //管道路径
    int _id;  //身份(user/creater)
    int _fd;  //文件描述符
};

server.cc 读端:

#include "namedPipe.hpp"

// server read:管理命名管道的生命周期(创建与删除)
int main()
{
    NamePiped fifo(comm_path, Creater);
    if(fifo.OpenForRead())
    {
        std::cout << "server open named pipe done" << std::endl;
        while (true)
        {
            std::string message;
            int res = fifo.ReadNamedpipe(&message);
            if (res > 0)//读内容
            {
                std::cout << "Client send >" << message << std::endl;
            }
            else if(res == 0)//写端关闭
            {
                std::cout << "Client quit" << std::endl;
                break;
            }
            else//出错
            {
                std::cout << "fifo.ReadNamedpipe default" << std::endl;
                break;
            }
        }
    }
    return 0;
}

运行:
在这里插入图片描述
光运行./server(读端),不运行写端会创建出管道,但读没有打印出server open named pipe done。说明没有打开管道。
在这里插入图片描述
然后运行了./client(写端)才会打开管道。说明对于读端而言,如果我们打开文件,但还没有写端,就会阻塞在open调用中,直到对方打开。简称进程同步。
在这里插入图片描述
相反如果读端关闭,写端还在写,写端就会收到SIGPIPE信号,让写端进程直接退出
在这里插入图片描述

共享内存

匿名管道和命名管道就是复用文件的代码。
有人从零开始写本地通信方案的代码:System V IPC 有三种方式通信
在这里插入图片描述
因为只能本地通信,且和文件的整合度不高,所以目前这种方案已经快被淘汰了。

原理

假设A进程在物理内存中创建一段内存空间,然后在A进程的地址空间中的共享区申请一片空间,再把虚拟地址与共享内存的映射关系填入页表。拥有映射关系后就可以往创建的内存中写入。B进程的虚拟地址也通过页表与创建的内存映射。这样进程A与进程B看到同一块资源,上面的技术叫做共享内存。
在这里插入图片描述
补充:
1.以上操作都是由操作系统做的,所以操作系统肯定提供给用户系统调用了。
2.AB进程通信,CD进程也要通信,所以共享进程在系统中可以存在很多份,且功能也不一样。操作系统就要对共享内存做管理!先描述,再组织。类似struct Shm的结构体。
在这里插入图片描述
综上所述:共享内存 = 共享内存空间(数据) + 共享内存属性

shmget

创建共享内存的函数调用:IPC_CREAT用于获取;IPC_CREAT | IPC_EXCL用于创建。
在这里插入图片描述
key比较特殊,是标识共享内存的唯一性字段,用于寻找共享空间。
在这里插入图片描述
综上:用户设置key值,AB两个进程就能看见同一块共享空间。
key值是给用户看的,后面会说shmid是给操作系统看的,所以操作系统不创建key值。
但是又不建议用户自己设置key,容易冲突,就给用户提供了函数ftok,由一些算法形成的随机数。

FIOK

同一个pathname,同一个proj_id,就能由算法形成同一个key。
在这里插入图片描述
返回值:成功后返回的是key的值;失败后返回-1
在这里插入图片描述

代码应用:

先验证共享内存的几个特性:
共享内存,不随着进程的结束释放。
Shm.hpp

#ifndef __SHM_HPP__
#define __SHM_HPP__

#include <iostream>
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/shm.h>
#include <cerrno>

#define gShmSize 4096

const std::string gpathname = "/root/test/shm";
const int gproj_id = 0x66;

std::string ToHex(key_t k)
{
    char buffer[128];
    snprintf(buffer,sizeof(buffer),"0x%x",k);
    return buffer;
}

key_t GetCommKey(const std::string pathname, int proj_id)
{
    key_t res = ftok(pathname.c_str(), proj_id);
    if (res < 0)
    {
        perror("create ftok failing");
    }
    return res;
}

int GetShm(int key)
{
    int shmid = shmget(key,gShmSize,IPC_CREAT | IPC_EXCL);
    if(shmid < 0)
    {
        perror("shmget fail");
    }
    return shmid;
}

#endif

server.cc

#include "Shm.hpp"

int main()
{
    key_t k = GetCommKey(gpathname,gproj_id);
    std::cout << "key: " << ToHex(k) << std::endl;
    int shmid = GetShm(k);
    std::cout << "shmid:" << shmid << std::endl;
    return 0;
}

运行两次,发现第二次并没有创建,并显示已经存在,说明共享内存并没有像子进程一样运行完就被父进程或系统回收。
在这里插入图片描述
把创建共享内存改成获取共享内存:

int GetShm(int key)
{
    int shmid = shmget(key,gShmSize,IPC_CREAT);
    if(shmid < 0)
    {
        perror("shmget fail");
    }
    return shmid;
}

发现获取的话可以一直获取。
在这里插入图片描述
如果不释放共享内存,就会一直存在,生命周期随内核。
查共享内存:

ipcs -m

在这里插入图片描述
key:属于用户形成,内核使用的一个字段,用户不能使用key来进行shm管理。
shmid:内核给用户返回的一个标识符,用来进行用户级对共享内存进行管理的id值。
删共享内存用命令shmid删除:

ipcrm -m [shmid]

在这里插入图片描述
也可以用操作系统提供的函数:shmctl删除。cmd删除是IPC_RMID,就是位图。
在这里插入图片描述
返回值:移植成功,返回0;失败,返回-1
在这里插入图片描述
综上所述:创建共享内存,使用后删除的代码就能写出来了:

#ifndef __SHM_HPP__
#define __SHM_HPP__

#include <iostream>
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/shm.h>
#include <unistd.h>
#include <cerrno>

#define gCreater 1
#define gUser 2
#define gShmSize 4096

const std::string gpathname = "/root/test/shm";
const int gproj_id = 0x66;

class Shm
{
private:
    key_t GetCommKey()
    {
        key_t key = ftok(_pathname.c_str(), _proj_id);
        if (key < 0)
        {
            perror("create ftok failing");
        }
        return key;
    }

    int GetShmHelper(key_t key, int size, int flag)
    {
        int shmid = shmget(key, size, flag);
        if (shmid < 0)
        {
            perror("shmget fail");
        }
        return shmid;
    }

public:
    Shm(const std::string pathname, const int proj_id, int who)
        : _pathname(pathname), _proj_id(proj_id), _who(who)
    {
        _key = GetCommKey();
        if (_who == gCreater)
        {
            CreaterGetShmid();
        }
        else if (_who == gUser)
        {
            UserGetShmid();
        }
        std::cout << "shmid: " << _shmid << std::endl;
        std::cout << "key: " << ToHex(_key) << std::endl;
    }

    ~Shm()
    {
        if (_who == gCreater)
        {
            int res = shmctl(_shmid, IPC_RMID, nullptr);
            if (res < 0)
                std::cout << "shmctl fail " << std::endl;
            else
                std::cout << "shmid remove done " << std::endl;
        }
    }

    std::string ToHex(key_t k)
    {
        char buffer[128];
        snprintf(buffer, sizeof(buffer), "0x%x", k);
        return buffer;
    }

    bool CreaterGetShmid()
    {
        if (_who == gCreater)
        {
            _shmid = GetShmHelper(_key, gShmSize, IPC_CREAT | IPC_EXCL);
            sleep(10); //为了创建完看见删除的效果
            if (_shmid > 0)
                return true;
        }
        std::cout << "Create shmid succeed" << std::endl;
        return false;
    }
    bool UserGetShmid()
    {
        if (_who == gUser)
        {
            _shmid = GetShmHelper(_key, gShmSize, IPC_CREAT);
            if (_shmid > 0)
                return true;
        }
        std::cout << "Get shmid succeed" << std::endl;
        return false;
    }

private:
    key_t _key;
    int _shmid;
    int _who;
    const std::string _pathname;
    const int _proj_id;
};

#endif

创建出共享内存后10秒钟后删除
在这里插入图片描述

prems

perms是共享内存的权限:如果创建的时候,加上权限,perms的值会更改。

bool CreaterGetShmid()
    {
        if (_who == gCreater)
        {
            _shmid = GetShmHelper(_key, gShmSize, IPC_CREAT | IPC_EXCL | 0666);
            if (_shmid > 0)
                return true;
        }
        std::cout << "Create shmid succeed" << std::endl;
        return false;
    }

在这里插入图片描述

nattch

nattch是该共享内存挂接的数量。
挂接的函数:shmat(at有attach的意思)
在这里插入图片描述
返回值:一旦挂接成功,返回地址空间中共享内存的起始地址。(跟malloc返回值类似)
挂接代码:

std::string RoleToString(int who)
    {
        if (who == gCreater)
            return "Creater";
        else if (who == gUser)
            return "gUser";
        else
            return "None";
    }
    // 挂接
    void *AttachShm()
    {
        void *shmaddr = shmat(_shmid, nullptr, 0);
        if (shmaddr == nullptr)
        {
            perror("shmat");
        }
        std::cout << "who: " << RoleToString(_who) << " attach shm..." << std::endl;
        return shmaddr;
    }

一个进程挂接上了为1,两个进程挂接上了为2。
在这里插入图片描述
去掉进程与共享内存的关联的函数:shmdt
在这里插入图片描述
代码如下:

void DetachShm(void *shmaddr)
    {
        if(shmaddr == nullptr) return;
        shmdt(shmaddr);
        std::cout << "who: " << RoleToString(_who) << " detach shm..." << std::endl;
    }

效果:0-》1-》0
在这里插入图片描述
上面都是准备工作:下面开始通信。
只运行读端:
现象:发现不等写端写入,读端一直在读。
在这里插入图片描述
读端和写端都运行:
现象:写端2秒写一次,读端1秒读1次,数据就有重复。(相比较通道读完数据就没有了)
在这里插入图片描述
读端和写端都运行,然后把写端关闭后再次运行写端:
现象:读端重新读取
在这里插入图片描述
综上所述:与管道不一样,共享内存不提供保护机制。
缺点:会造成数据不一致问题(在纸上写字,写一半被别人拿走了)
优点:共享内存收所有进程IPC,速度最快的。因为共享内存大大减少了数据的拷贝次数。
在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2231353.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

基于SSM+小程序的计算机实验室排课与查询管理系统(实验室2)

&#x1f449;文末查看项目功能视频演示获取源码sql脚本视频导入教程视频 1、项目介绍 1、管理员功能有个人中心&#xff0c;学生管理&#xff0c;教师管理&#xff0c;实验室信息管理&#xff0c;实验室预约管理&#xff0c;取消预约管理&#xff0c;实验课程管理&#xff0…

基于STM32的农业监测与管理系统设计思路介绍(代码示例)

一、项目概述 在全球农业现代化进程中&#xff0c;农业监测与管理系统的研发具有重要意义。本文介绍的基于STM32的农业监测与管理系统&#xff0c;旨在通过智能小车实现对农作物的环境监测、土壤检测等功能。该系统利用手势控制技术&#xff0c;农民可以通过简单的手势指令来操…

分布式架构搭建博客网站

目录 运行环境基础配置需求准备工作配置静态ip修改主机名及host映射开启防火墙时间同步配置免密ssh登录 环境搭建Server-Web端安装LNMP环境软件Server-NFS-DNS端上传博客软件Server-NFS-DNS端设置NFS共享Server-Web设置挂载远程共享目录nginx设置在数据库中创建数据库和用户重启…

基于Transformer的路径规划 - 第五篇 GPT生成策略_解码方法优化

上一篇&#xff1a;基于Transformer的路径规划 - 第四篇 GPT模型优化 在上一篇中&#xff0c;我尝试优化GPT路径生成模型&#xff0c;但没有成功。在随机生成的测试集上&#xff0c;路径规划成功率只有99%左右。而使用传统的路径规划算法&#xff0c;例如A*&#xff0c;路径规划…

【HarmonyOS】鸿蒙应用设置控件通用样式AttributeModifier, @Styles

【HarmonyOS】鸿蒙应用设置控件通用样式AttributeModifier&#xff0c; Styles 前言 在鸿蒙中UI开发经常需要对控件样式进行统一的封装&#xff0c;在API早前版本&#xff0c;一般是通过 Styles进行样式封装复用&#xff1a; Entry Component struct Index {build() {Column(…

[vulnhub]DC:7

https://www.vulnhub.com/entry/dc-7,356/ 端口扫描主机发现 探测存活主机&#xff0c;178是靶机 nmap -sP 192.168.75.0/24 Starting Nmap 7.94SVN ( https://nmap.org ) at 2024-11-03 13:30 CST Nmap scan report for 192.168.75.1 Host is up (0.00037s l…

鸿蒙HarmonyOS应用开发者(基础+高级)认证

文章目录 鸿蒙HarmonyOS应用开发者(基础高级)认证&#x1f449;1.HarmonyOS认证介绍1.1、HarmonyOS发展历程1.2、HarmonyOS NEXT 开发预览版1.3、ArkTS语言开发鸿蒙应用1.4、HarmonyOS应用开发者基础认证的核心内容1.5、HarmonyOS应用开发者高级认证的核心内容1.6、HarmonyOS应…

视频Qoe测量学习笔记(一)

目录 流媒体协议详解 RTSP&#xff1a;实时流式协议 RTCP&#xff1a;实时运输控制协议 RTP&#xff1a;实时运输协议 H.264 流媒体协议详解 RTSP&#xff1a;实时流式协议 由IETF MMusic小组开发&#xff0c;已成为互联网建议标准[RFC 2326]。RTSP本身并不传送数据&…

第三十三章 Vue路由进阶路由模块封装

目录 一、引言 二、完整代码 main.js index.js App.vue Find.vue My.vue 一、引言 在上一个章节中&#xff0c;我们将所有的路由配置都堆在main.js中来实现路径组件的路由&#xff0c;这样做的话非常不利于我们后期对项目的维护。因此正确的做法是将路由模块抽离出来&a…

用插值公式实现滚动进度条动画效果

我们在日常前端开发时在动画的选择上基本都是css&#xff0c;通过css的animation即可满足大部分的开发场景&#xff0c;如果遇到了特殊而比较不容易实现的效果就会考虑到用js来实现&#xff0c;而本次的主题&#xff0c;就是围绕用js来做一个比较不常见的特殊动画效果。 假设我…

【uni-app】创建自定义模板

1. 步骤 打开自定义模板文件夹 在此文件夹下创建模板文件&#xff08;注意后缀名&#xff09; 重新点击“新建页面” 即可看到新建的模板 2. 注意事项 创建的模板必须文件类型对应&#xff08;vue模板就创建*.vue文件, uvue模板就创建*.uvue文件&#xff09;

03哈希表算法/代码随想录

三、哈希表 有效的字母异位词 力扣242 这题是典型的哈希映射&#xff0c;只要将t存到哈希表中&#xff0c;key为t拆解的值&#xff0c;value为t中有过个key这样的值&#xff0c;然后在使用哈希表O&#xff08;1&#xff09;的时间复杂度判断 class Solution {public boolean …

下载安装COPT+如何在jupyter中使用(安装心得,windows,最新7.2版本)

目录 1.到杉树科技官网申请下载COPT 2.安装COPT&配置许可文件 3.在jupyter中使用COPT的python接口 最近看到一本和数学建模有关的新书&#xff1a;《数学建模与数学规划&#xff1a;方法、案例及编程实战》&#xff0c;作为数学建模老手&#xff0c;肯定要学习一下&…

【Linux】——操作系统-进程详解

大家好呀&#xff0c;我是残念&#xff0c;希望在你看完之后&#xff0c;能对你有所帮助&#xff0c;有什么不足请指正&#xff01;共同学习交流哦 本文由&#xff1a;残念ing原创CSDN首发&#xff0c;如需要转载请通知 个人主页&#xff1a;残念ing-CSDN博客&#xff0c;欢迎各…

Excel:vba实现批量插入图片批注

实现的效果&#xff1a;实现的代码如下&#xff1a; Sub InsertImageNamesAndPictures()Dim PicPath As StringDim PicName As StringDim PicFullPath As StringDim RowNum As IntegerDim Name As StringDim Comment As CommentDim folder As FileDialog 定义文件选择对话框 清…

HTML 语法规范——代码注释、缩进与格式、标签与属性、字符编码等

文章目录 一、代码注释1.1 使用注释的主要目的1.2 使用建议二、标签的使用2.1 开始标签和结束标签2.2 自闭合标签2.3 标签的嵌套2.4 标签的有效性三、属性四、缩进与格式4.1 一致的缩进4.2 元素单独占用一行4.3 嵌套元素的缩进4.4 避免冗长的行五、字符编码六、小结在开发 HTML…

闯关leetcode——242. Valid Anagram

大纲 题目地址内容 解题代码地址 题目 地址 https://leetcode.com/problems/valid-anagram/ 内容 Given two strings s and t, return true if t is an anagram of s, and false otherwise. Example 1: Input:s “anagram”, t “nagaram” Output:true Example 2: Inp…

无人机之远程指挥中心技术篇

一、核心功能 实时监控与控制&#xff1a; 通过高清视频流和其他传感器数据&#xff0c;指挥中心可以实时了解无人机的当前状态、位置和环境情况。操作人员可以在指挥中心对无人机进行精确的飞行控制&#xff0c;包括起飞、降落、悬停、移动等&#xff0c;确保无人机按照预定…

C++学习路线(数据库部分)二

类型 整形类型 整数类型是数据库中最基本的数据类型。标准SQL中支持INTEGER和SMALLINT这两种数据类型。MySQL数据库除了支持这两种类型以外&#xff0c;还扩展支持了TINYINT、MEDIUMINT和BIGINT。下表从不同整数类型的字节数、取值范围等方面进行对比。 类型名称后面的小括号…

秒杀优化(异步秒杀,基于redis-stream实现消息队列)

目录 秒杀优化一&#xff1a;异步秒杀1&#xff1a;思路2&#xff1a;实现 二&#xff1a;redis实现消息队列1&#xff1a;什么是消息队列2&#xff1a;基于list结构实现消息队列3&#xff1a;基于pubsub实现消息队列4&#xff1a;基于stream实现消息队列5&#xff1a;stream的…