Linux 进程间通信之共享内存

💓博主CSDN主页:麻辣韭菜💓

⏩专栏分类:Linux知识分享⏪

🚚代码仓库:Linux代码练习🚚

🌹关注我🫵带你学习更多Linux知识
  🔝 


目录

​编辑​

前言

共享内存直接原理 

共享内存数据结构 

 共享内存函数接口

 shmget

函数原型

参数说明:

返回值:

系统限制:

额外说明:

ftok

shmat

shmdt

函数原型

参数说明

返回值

附加说明

错误代码

shmctl

 代码实现

获得key 

创建共享内存 

 挂接

 取消挂接

 删除共享内存

 共享内存命令行操作


前言

前面讲的匿名和命名都是基于文件看到同一份资源,那有没有不通过文件可以看到同一份资源?有的系统 V

共享内存直接原理 

  • 每个进程都有属于自己的进程地址空间。
  • 操作系统在物理内存中开辟一段空间,一个进程创建一段虚拟内存,并将这段虚拟内存的起始与结束地址通过页表映射到物理内存的空间。
  • 如果另一个进程也通过页表映射到同一段物理内存,那么就可以实现了让多个进程看到同一段空间,当一个进程向这段物理空间写入数据,另一个进程就可以马上从这段空间读取数据,实现进程间的通信。

共享内存需不需要被管理?当然需要啊,又不是只有这一个共享内存,其他进程万一也需要通信。共享内存多了当然需要管理。那怎么管理?老规矩先描述再组织!!!

共享内存数据结构 

struct shmid_ds {
        struct ipc_perm shm_perm;         /* operation perms */
        int shm_segsz;         /* size of segment (bytes) */
        __kernel_time_t shm_atime;         /* last attach time */
        __kernel_time_t shm_dtime;         /* last detach time */
        __kernel_time_t shm_ctime;         /* last change time */
        __kernel_ipc_pid_t shm_cpid;         /* pid of creator */
        __kernel_ipc_pid_t shm_lpid;         /* pid of last operator */
        unsigned short shm_nattch;         /* no. of current attaches */
        unsigned short shm_unused;         /* compatibility */
        void *shm_unused2;         /* ditto - used by DIPC */
        void *shm_unused3;         /* unused */
};

描述出来以后,对共享内存的管理,变成了对shmid_ds增删查改!!! 

 共享内存函数接口

 shmget

shmget 函数是 Linux 系统中用于创建或获取共享内存段标识符的函数。以下是 shmget 函数的详细解析:

函数原型

#include <sys/ipc.h>

#include <sys/shm.h>

int shmget(key_t key, size_t size, int shmflg);

参数说明:

  1. key:一个标识符,用于唯一确定系统中的共享内存段。它可以是:

    • IPC_PRIVATE:一个特殊的键值,用于创建新的共享内存段。
    • 由 ftok 函数生成的键值,通常基于文件名和项目 ID。
  2. size:要创建的共享内存段的大小,以字节为单位。实际创建的共享内存大小将是 size 向上取最近的 PAGE_SIZE(页面大小)的倍数。

  3. shmflg:共享内存的访问权限和控制标志,可以是以下选项的组合:

    • IPC_CREAT:如果共享内存不存在,则创建它。
    • IPC_EXCL:与 IPC_CREAT 一起使用,如果共享内存已存在,则 shmget 调用失败。
    • SHM_R 或 SHM_RDONLY:共享内存段为只读。
    • SHM_W:共享内存段为可写。

返回值:

  • 成功:返回共享内存段的标识符(shmid)。
  • 失败:返回 -1,并设置全局变量 errno 以指示错误类型。

错误代码:

  • EINVALsize 小于 SHMMIN 或大于 SHMMAX
  • EEXISTshmflg 包含 IPC_CREAT | IPC_EXCL,但已存在与 key 相匹配的共享内存。
  • EIDRM:尝试获取的共享内存已经被删除。
  • ENOENTkey 不存在,且 shmflg 没有设置 IPC_CREAT
  • EACCES:没有足够的权限访问指定的共享内存。

系统限制:

  • SHMALL:系统范围内共享内存段的总数限制。
  • SHMMAX:单个共享内存段的最大大小限制。
  • SHMMNI:系统范围内共享内存段标识符的最大数量限制。

额外说明:

  • 使用 shmget 创建共享内存时,通常会结合 shmctlshmat 和 shmdt 函数来完成共享内存的控制、映射、使用和脱离。
  • IPC_PRIVATE 作为 key 时,会忽略 shmflg 的低9位,并创建一个新的共享内存段。

ftok

ftok 是一个UNIX系统调用,用于创建一个唯一的键值,这个键值可以用于ftok函数创建的两个进程之间的通信。它通常用于初始化或识别一个共享内存段或信号量。

函数原型如下:

key_t ftok(const char *pathname, char proj_id);

参数说明:

  • pathname:一个字符串,指定了一个文件的路径。这个文件的存在并不是必需的,但是它必须存在于文件系统中。
  • proj_id:一个字符,用于进一步区分同一个文件下生成的不同键值。

返回值:

  • 成功时返回一个key_t类型的键值,这个键值可以用于IPC(进程间通信)。
  • 失败时返回-1,并设置errno以指示错误类型。

ftok函数通过文件名和项目标识符生成一个键值,这个键值可以用来创建或访问共享内存段或信号量。ftok函数的实现通常依赖于文件的设备号和文件名的i节点号,以及项目标识符,来生成一个唯一的键值。

请注意,ftok函数是POSIX标准的,主要用于UNIX和类UNIX系统。在Windows系统中,该函数不可用。


shmat

shmat 是一个用于进程间通信(IPC)的系统调用函数,它是System V共享内存接口的一部分。shmat 函数用于将一个已经创建的共享内存段附加(attach)到调用进程的地址空间中。一旦附加,进程就可以像访问普通内存一样访问共享内存。

函数原型如下:

void *shmat(int shm_id, const void *shmaddr, int shmflg);

参数说明:

  1. shm_id:共享内存段的标识符,通常由 shmget 函数创建并返回。
  2. shmaddr:(可选)一个指针,指定了共享内存段在调用进程地址空间中附加的位置。如果设置为NULL,系统会找到一个合适的地址来附加共享内存。
  3. shmflg:控制共享内存附加行为的标志。常见的标志有:
    • SHM_RDONLY:以只读方式附加共享内存。
    • 0:以读写方式附加共享内存。

返回值:

  • 成功时返回一个指向共享内存段的指针,进程可以使用这个指针来访问共享内存。
  • 失败时返回-1,并设置全局变量 errno 以指示错误类型。

使用 shmat 时,需要确保已经包含了相应的头文件,如 <sys/ipc.h><sys/shm.h>


shmdt

shmdt 函数是用于断开与共享内存段的连接的Linux系统调用。当一个进程通过shmat函数将共享内存段映射到自己的地址空间后,如果不再需要访问该共享内存段,就可以使用shmdt函数来断开这种映射关系。以下是shmdt函数的一些关键信息:

函数原型

int shmdt(const void *shmaddr);

参数说明

  • shmaddr:共享内存段连接到当前进程地址空间的起始地址。

返回值

  • 成功:返回0。
  • 出错:返回-1,并且errno会被设置为相应的错误代码。

附加说明

  • 本函数调用并不删除所指定的共享内存段,而只是将先前用shmat函数连接(attach)好的共享内存脱离(detach)当前进程。
  • 如果shmaddr参数无效,将返回错误。

错误代码

  • EINVAL:无效的参数shmaddr

 


shmctl

shmctl 是一个用于进程间通信(IPC)的系统调用函数,它是System V共享内存接口的一部分。shmctl 函数用于控制共享内存段的各种操作,如获取共享内存段的信息、设置共享内存段的属性或删除共享内存段。

函数原型如下:

int shmctl(int shm_id, int cmd, struct shmid_ds *buf);

参数说明:

  1. shm_id:共享内存段的标识符,通常由 shmget 函数创建并返回。
  2. cmd:一个命令,指定了要执行的操作。常见的命令有:
    • IPC_STAT:获取共享内存段的状态信息,并将信息存储在 buf 指向的结构体中。
    • IPC_SET:设置共享内存段的属性,buf 指向的结构体包含了新的属性值。
    • IPC_RMID:删除共享内存段,释放相关资源。
  3. buf:一个指向 shmid_ds 结构体的指针,该结构体用于存储共享内存段的信息或新的属性设置。

返回值:

  • 成功时返回0。
  • 失败时返回-1,并设置全局变量 errno 以指示错误类型。

shmid_ds 结构体通常包含以下成员:

  • shm_perm:共享内存段的权限。
  • shm_segsz:共享内存段的大小。
  • shm_atime:最后访问时间。
  • shm_dtime:最后删除时间。
  • shm_ctime:最后修改时间。
  • shm_cprid:最后执行读操作的进程的进程ID。
  • shm_lprid:最后执行写操作的进程的进程ID。

使用 shmctl 时,需要确保已经包含了相应的头文件,如 <sys/ipc.h><sys/shm.h>

 代码实现

我们要创建共享内存,首先要有key而key是调用ftok来获得,我们先创建一个comm.hpp

创建共享内存的需要的用到的系统调,我们直接用这个头文件进行封装,然后prosessa 和 prosessb直接进行调用

获得key 

 

#include <iostream>
#include <string>
#include <sys/ipc.h>
#include <sys/shm.h>

using namespace std;
const string pathname = "/home/gx/linux-exercise";
const int proj_id = 0x666; // 这个你可以自己随便定一个。

key_t get_key()
{
    key_t key = ftok(pathname.c_str(), proj_id);
    if (key == -1)
    {
        cerr << "ftok error" << endl;
        exit(1);
    }
    return key;

}

创建共享内存 

 

int create_shm(int flag)
{
    key_t key = get_key();
    int n = shmget(key, SHM_SIZE, flag);
    if(n < 0)
    {
        cerr << "shmget error" << endl;
        exit(1);        
    } 
    return n;                    
}

这里我用a进程进行创建共享内存,b获得共享内存,这样的话对create_shm函数进行封装,a调用createShm这个函数。

int createShm()
{
    return create_shm(IPC_CREAT | IPC_EXCL | 0666);
}

b就调用getShm这个函数。 

int getShm()
{
    return create_shm(IPC_CREAT);
}

关于shmget的shmflg这个参数,前面文档已经有详细的说明。这里不在细说。

共享内存创建好了,就意味着我们能通信了吗?并没有,因为shmget是系统调用,是OS划分出来的一块区域。 是真实的物理内存,而进程用的是虚拟地址,我们需要用shmat进行挂接。通过挂接,挂接到进程地址空间。        

 挂接

 char* shmaddr = (char*)shmat(shmid,nullptr,0);

 取消挂接

 shmdt(shmaddr);

 删除共享内存

 shmctl(shmid, IPC_RMID, nullptr);

 

完整代码  

头文件

#include <iostream>
#include <string>
#include <cstring>
#include <cstdlib>
#include <cstdio>
#include <sys/ipc.h>
#include <sys/shm.h>
#include <unistd.h>


using namespace std;
const int SHM_SIZE = 4096; // 共享内存的大小 也就是4KB,我们申请的大小一般是4096的整数倍
const string pathname = "/home/gx/linux-exercise";
const int proj_id = 0x666; // 这个你可以自己随便定一个,只要保证唯一就行

key_t get_key()
{
    key_t key = ftok(pathname.c_str(), proj_id);
    if (key == -1)
    {
        cerr << "ftok error" << endl;
        exit(1);
    }
    return key;
}

int create_shm(int flag)
{
    key_t key = get_key();
    int n = shmget(key, SHM_SIZE, flag);
    if (n < 0)
    {
        cerr << "shmget error" << endl;
        exit(1);
    }
    return n;
}

int createShm()
{
    return create_shm(IPC_CREAT | IPC_EXCL | 0666);
}
int getShm()
{
    return create_shm(IPC_CREAT);
}

进程a 

 

int main()
{
    int shmid = createShm();
    char *shmaddr = (char *)shmat(shmid, nullptr, 0);

    while (true)
    {

        cout << "Enter a message: " ;
        fgets(shmaddr, 4096, stdin);
        
    }

    shmdt(shmaddr);
    shmctl(shmid, IPC_RMID, nullptr);

    return 0;
}

 进程b

 

int main()
{
    int shmid = getShm();
    char* shmaddr = (char*)shmat(shmid,nullptr,0);
    while(true)
    {
        cout << "client say@:" << shmaddr << endl;
        sleep(1);
    }
    shmdt(shmaddr);

    return 0;   
}

代码运行起来,我们可以发现共享内存是不会同步的,一方不写,一方也不会阻塞。

有没有什么方法可以让它们同步互斥?用管道。 

 共享内存命令行操作

查看共享内存

指令:ipcs -m 

 

删除共享内存

指令:ipcrm -m shmid 

 

 最后说说 key 和shmid的区别 key是操作系统创建出来的,确定这个共享内存的唯一性。

而shmid是给用户用的。

 

 

 

 

 

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

使用双指针解决问题题集(二)

1. 有效三角形的个数 给定一个包含非负整数的数组 nums &#xff0c;返回其中可以组成三角形三条边的三元组个数。 示例 1: 输入: nums [2,2,3,4] 输出: 3 解释:有效的组合是: 2,3,4 (使用第一个 2) 2,3,4 (使用第二个 2) 2,2,3 示例 2: 输入: nums [4,2,3,4] 输出: 4 题解&a…

zabbix监控方式(zabbix-trapper)

中文&#xff1a;zabbix采集器&#xff0c;即zabbix sender 。 Zabbix-Trapper 监控方式可以一次批量发送数据给Zabbix Server&#xff0c;与主动模式不同&#xff0c;Zabbix-Trapper 可以让用户控制数据的发送&#xff0c;而不用Zabbix-Agent进程控制&#xff0c;这意味着可以…

Angular中的管道(Pipe)

Angular中的管道(Pipe) 文章目录 Angular中的管道(Pipe)前言一、内置管道1. date管道格式化日期2. currency管道格式化货币3. uppercase和lowercase管道转换字符串大小写4. 小数位数5. JavaScript 对象序列化6. slice7. 管道链 二、自定义管道 前言 Angular中的管道&#xff0…

力扣刷题--数组--第二天

今天仍然做二分查找相关的题目。先来回顾一下二分查找的方法和使用的条件。二分查找是在数组中查找目标值的一种方法&#xff0c;通过边界索引确定中间索引&#xff0c;判断中间索引处的元素值和目标值的大小&#xff0c;来不断缩小查找区间。使用二分查找有如下一些限制&#…

ASP.NET通用作业批改系统设计

摘  要 该系统采用B/S结构&#xff0c;以浏览器方式登陆系统&#xff0c;用ASP.NET作为开发语言&#xff0c;数据库则使用Microsoft SQL Server 2000实现。《通用作业批改系统》包括了学生子系统、教师子系统、管理员子系统三大模块&#xff0c;该系统主要完成学生&#xff…

LibTorch入坑记--续2

一、安装faiss 我的faiss&#xff0c;用的是曾经安装过的 pip install faiss-gpu1.7 当时搞得环境名称是pni 二、配置环境 三、例子代码 #include <faiss/IndexFlat.h> #include <faiss/Index.h> #include <faiss/VectorTransform.h> #include <faiss/…

Mybatis进阶4-权限管理

权限管理 1.权限 //相当于 职责 2.用户 //相当于 职员&#xff08;职员就职于一个职位&#xff09; 3.角色 //相当于 职位&#xff08;有多个职责&#xff09; 权限管理基础表&#xff1a;权限表&#xff0c;用户表&#xff0c;角色表 问题1&#xff1a;…

SVM直观理解

https://tangshusen.me/2018/10/27/SVM/ https://www.bilibili.com/video/BV16T4y1y7qj/?spm_id_from333.337.search-card.all.click&vd_source8272bd48fee17396a4a1746c256ab0ae SVM是什么? 先来看看维基百科上对SVM的定义: 支持向量机&#xff08;英语&#xff1a;su…

根据最近拒包项目总结,详细讲解Google最新政策(上)

关于占比最多的移动垃圾软件拒审问题 移动垃圾软件(Mobile Unwanted Software)特征表现1> 具有欺骗性,承诺其无法实现的价值主张。2> 诱骗用户进行安装,或搭载在用户安装的其他程序上。3> 不向用户告知其所有主要功能和重要功能。4> 以非预期方式影响用户的系统…

Error Code: 1449. The user specified as a definer (‘admin‘@‘%‘) does not exist

前言 在进行MySQL数据库迁移或存储过程部署时&#xff0c;您可能会遇到错误 [Err] 1449 - The user specified as a definer (admin%) does not exist。这篇文章将为您提供一个详细的解决方案&#xff0c;帮助您顺利解决这一问题。 错误背景 此错误通常发生在尝试执行一个存…

Jenkins集成Kubernetes 部署springboot项目

文章目录 准备部署的yml文件Harbor私服配置测试使用效果Jenkins远程调用参考文章 准备部署的yml文件 apiVersion: apps/v1 kind: Deployment metadata:namespace: testname: pipelinelabels:app: pipeline spec:replicas: 2selector:matchLabels:app: pipelinetemplate:metada…

机器学习算法--朴素贝叶斯(Naive Bayes)

一、实验环境 1. python3.7 2. numpy > 1.16.4 3. sklearn > 0.23.1 二、朴素贝叶斯的介绍 朴素贝叶斯算法&#xff08;Naive Bayes, NB) 是应用最为广泛的分类算法之一。它是基于贝叶斯定义和特征条件独立假设的分类器方法。NB模型所需估计的参数很少&#xff0c;对缺…

【微服务】网关(详细知识以及登录验证)

微服务网关 网关网关路由快速入门路由属性 路由断言网关登录校验自定义过滤器实现登录校验网关传递用户OpenFeign传递用户 网关 网络的关口&#xff0c;负责请求的路由&#xff0c;转发&#xff0c;身份校验 当我们把一个单体项目分成多个微服务并部署在多台服务器中&#xff…

DDR4 新功能介绍

DDR4(第四代双倍数据率同步动态随机存取内存)相较于其前代DDR3,引入了一些新的功能和改进,这些新功能有助于提高内存的性能、降低功耗以及增强系统的可靠性,包括VPP、DBI(Data Bus Inversion,数据总线翻转)和DMI(与LPDDR4相关)。以下是对这些功能的简要说明: 更高的…

java—异常

异常 什么是异常 异常的体系 编译时异常处理方式 1、选择报错的整个代码块&#xff0c;快捷键crtlaltt键&#xff0c;选择try/catch将代码围起来。 2、编译异常处理方式2 在main方法上抛出异常 自定义异常 例子&#xff1a; 自定义运行时异常 自定义编译时异常 异常…

视觉图像信息处理与FPGA实现第九次作业——直方图均衡

RAM的B站视频解析 RAM的文档 一、65536x8位的单端口RAM timescale 1ns / 1ps //SPRF Single Port Read/Write Function //65535 是RAM中总的字数&#xff0c;也就是存储深度&#xff0c;X8表示每个字是8位的 module SPRF65536X8(Q,CLK,CEN,WEN,A,D );//输出寄存器Qoutput [7…

如何在已经安装好的PostgreSQL14中安装uuid 扩展

当前环境 PG14.8 LINUX 8.8 存在问题&#xff1a; 开发人员问&#xff0c;PG中&#xff0c;支持 生成UUID吗&#xff0c;具体是什么&#xff0c;答&#xff0c;类似这个函数 uuid_generate_v4() 看了一下&#xff0c; select uuid_generate_v4();会报错&#xff0…

2024-05-07 商业分析-如何在社会层面做一个更好的工具人-记录

摘要: 2024-05-07 商业分析-如何成为一个靠谱的工具人 如何在社会层面做一个更好的工具人 那么今天讲的这个主题呢&#xff0c;对吧&#xff1f;你们一看啊&#xff0c;就觉得这个就不应该我讲是吧啊&#xff0c;但是呢这个逻辑呢我还得跟你们讲一下啊&#xff0c;就是如何成为…

2009-2022年上市公司华证ESG评级评分数据(含细分项)

2009-2022年上市公司华证ESG评级评分数据&#xff08;含细分项&#xff09; 1、时间&#xff1a;2009-2022年 2、来源&#xff1a;华证ESG 3、指标&#xff1a;证券代码、证券简称、综合评级、年度、综合得分、E评级、E得分、S评级、S得分、G评级、G得分 4、范围&#xff1…

AI伦理和安全风险管理终极指南

人工智能&#xff08;AI&#xff09;正在迅速改变各个领域的软件开发和部署。驱动这一转变的两个关键群体为人工智能开发者和人工智能集成商。开发人员处于创建基础人工智能技术的最前沿&#xff0c;包括生成式人工智能&#xff08;GenAI&#xff09;模型、自然语言处理&#x…