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是给用户用的。

 

 

 

 

 

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

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

相关文章

Spring Boot3.x集成Disruptor4.0

Disruptor介绍 Disruptor是一个高性能内存队列&#xff0c;研发的初衷是解决内存队列的延迟问题(在性能测试中发现竟然与I/O操作处于同样的数量级)。基于Disruptor开发的系统单线程能支撑每秒600万订单&#xff0c;2010年在QCon演讲后&#xff0c;获得了业界关注。2011年&…

Oracle12之后json解析包怎么调用

在 Oracle 12g 及之后的版本中&#xff0c;Oracle 提供了对 JSON 的原生支持&#xff0c;使得在数据库中存储、查询和解析 JSON 数据变得更为简单。你可以使用 Oracle 提供的 SQL 函数和操作符来处理 JSON 数据。 以下是一些常用的 Oracle SQL 函数和操作符&#xff0c;用于解…

02-大厂电商设计解析之商品管理系统

1 雪花算法使用 IdWorker idWorkernew IdWorker(1,1); for(int i0;i<10000;i){long id idWorker.nextId();System.out.println(id); } 配置分布式ID生成器 将IdWorker.java拷贝到util包在工程的resources下新增applicationContext-service.xml <!‐‐雪花ID生成器‐…

geojson文件规格

geojson文件示例&#xff0c; {"type": "FeatureCollection","features": [{"type": "Feature","geometry": {"type": "Point","coordinates": [102.0, 0.5]},"properties&q…

阴影渲染在AI去衣技术中的关键作用

引言&#xff1a; 随着人工智能技术的飞速发展&#xff0c;深度学习在图像处理领域取得了突破性的进展。其中&#xff0c;AI去衣技术作为一种高度复杂的图像到图像的转换过程&#xff0c;不仅要求算法能够精确地识别并处理衣物纹理和结构&#xff0c;还要求生成的结果具有高度的…

SaaS应用加速解决方案

随着企业业务的迅速扩展&#xff0c;SaaS应用成为企业提升办公效率的关键。然而&#xff0c;在SaaS应用广泛使用的同时&#xff0c;访问速度受限、网络拥堵等问题也逐渐浮现。为了解决这些挑战&#xff0c;SaaS应用加速方案应运而生&#xff0c;旨在助力企业高效运转&#xff0…

【网络】IP层分片和TCP分段,有MTU为什么还需要MSS

引言 在网络通信中&#xff0c;IP层分片和TCP分段是两个重要的概念&#xff0c;它们分别在网络层和传输层发挥着关键作用。本文将介绍IP层分片的作用和缺陷&#xff0c;解释为何TCP需要分段而不是让IP层进行分片。 IP层分片&#xff1a;作用和缺陷 作用&#xff1a; IP层分…

基于51单片机ESP8266wifi控制机器人—送餐、快递

基于51单片机wifi控制机器人 &#xff08;程序&#xff0b;原理图&#xff0b;PCB&#xff0b;设计报告&#xff09; ​功能介绍 具体功能&#xff1a; 1.L298N驱动电机&#xff0c;机器人行走&#xff1b; 2.装备红外线感应检测到周围环境&#xff0c;进行行程判断&#xf…

Tensorflow-相关函数

边学边查,后续不断补充完善 1、tf.placeholder() 函数形式: tf.placeholder(dtype,shape=none,name=none) dtype:数据类型,常用的是tf.foat32,tf.float64等数值类型 shape:数据形状,默认是none,就是一堆值,也可以是多维(比如[2,3],[none,3]表示列上3,行不定) …

C++ 顺序线性表的功能

顺序线性表的功能 //头文件 #pragma once#define LIST_INIT_SIZE 50 #define LIST_INCREMENT 20 #define OK 1 #define ERROR 0typedef int Status;typedef char ElemType;typedef struct list_ {ElemType* elem;int length;int listize; }SqList;// 1 初始化 函数 Status…

基于单片机的太阳能无线手机充电器的设计

摘 要:为了解决手机户外无法充电而不能正常使用的问题,设计一款基于单片机的便携式太阳能无线智能手机充电器,利用太阳能电池板将光能转换为电能,由降压稳压电路将电能存储于蓄电池中,通过无线电力传输模块将电能传输至手机终端,由降压稳压处理后给手机充电,并结合单片机…

在Ubuntu上搭建幻兽帕鲁服务器

简介 幻兽帕鲁是一款多人在线角色扮演游戏&#xff0c;玩家可以捕捉和训练各种各样的幻兽&#xff0c;并与其他玩家进行战斗和交易。如果您想拥有自己的幻兽帕鲁服务器&#xff0c;可以按照以下步骤在 Ubuntu 上进行搭建。 准备工作 在开始之前&#xff0c;您需要准备以下几…

代码随想录第52天|300.最长递增子序列 718. 最长重复子数组

300.最长递增子序列 300. 最长递增子序列 - 力扣&#xff08;LeetCode&#xff09; 代码随想录 (programmercarl.com) 动态规划之子序列问题&#xff0c;元素不连续&#xff01;| LeetCode&#xff1a;300.最长递增子序列_哔哩哔哩_bilibili 给你一个整数数组 nums &#xff0…

OPENAI中Semantic Kernel的优点和缺点

OPENAI中Semantic Kernel的优点和缺点 1. 引言 Semantic Kernel是OPENAI中的一项重要技术&#xff0c;它在自然语言处理和语义理解方面具有广泛的应用。本文将探讨Semantic Kernel的优点和缺点&#xff0c;并提出在项目中如何避免这些缺点的方法。 2. Semantic Kernel的优点…

multipass和multipassd命令的区别

multipassd通常是multipass服务的后台守护进程&#xff0c;它负责管理和控制虚拟机实例。 命令区别 例&#xff1a; multipass restart my-vm 这个命令用于重启Multipass中的虚拟机实例。例如有一个名为my-vm的虚拟机实例。 multipassd restart 这会重新启动Multipass后台…

Scroll生态项目Penpad,再获Presto Labs的投资

Penpad是Scroll生态的LaunchPad平台&#xff0c;其整计划像收益聚合器以及RWA等功能于一体的综合性Web3平台拓展&#xff0c;该平台在近期频获资本市场关注&#xff0c;并获得了多个知名投资者/投资机构的支持。 截止到本文发布前&#xff0c;Penpad已经获得了包括Scroll联合创…

了解 Postman:这个 API 工具的功能和用途是什么?

在软件开发中&#xff0c;经常听到 Postman 这个软件名。但其实很多新手开发者只知道这是软件开发常用的软件&#xff0c;并不知道实际是一个什么样工具&#xff0c;不知道具体的作用是什么。那今天就跟大家好好唠唠 Postman 这个软件。想要学习更多关于 Postman 的知识&#x…

Codigger:Web应用赋能的分布式操作系统让用户卓越体验

Codigger&#xff0c;作为一个分布式操作系统&#xff0c;其独特之处在于其采用的浏览器/服务器&#xff08;Browser/Server&#xff0c;简称B/S&#xff09;架构。这种架构的核心思想是&#xff0c;通过浏览器来进入工作界面&#xff0c;页面交互部分事务逻辑在前端&#xff0…

最小化蒙德城的旅行者队伍(巴士)

描述 在阳光明媚的一天&#xff0c;凯亚在蒙德城的风车塔下等待着前往狼的领域的旅行者。他于12:00抵达&#xff0c;并计划在此地逗留一整小时&#xff0c;直至12:59。 蒙德城有许多旅行者的车队&#xff0c;每个车队都有自己的出发时间表。 凯亚观察了这些车队的出发时间&a…

【Linux网络】PXE批量网络装机

目录 一、系统装机 1.1 三种引导方式 1.2 系统安装过程 1.3 四大重要文件 二、PXE 2.1 PXE实现原理 2.2 PXE手动搭建过程 2.3 kickstart配合pxe完成批量自动安装 一、系统装机 1.1 三种引导方式 硬盘光驱(U盘)网络启动 1.2 系统安装过程 加载boot loader加载启动安…