Linux多进程(二)进程通信方式三 共享内存

news2024/11/27 10:43:45

共享内存提供了一个在多个进程间共享数据的方式,它们可以直接访问同一块内存区域,因此比使用管道或消息队列等通信机制更高效。在多进程程序中,共享内存通常与信号量一起使用,以确保对共享内存的访问是线程安全的。

一、打开/创建共享内存

shmget系统调用创建一段新的共享内存,或者获取一段已经存在的共享内存。其定义如下:

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

int shmget(key_t key, size_t size, int shmflg);
  • key: 共享内存标识符,可以由 ftok 函数生成或使用 IPC_PRIVATE
  • size: 要创建的共享内存段的大小(以字节为单位)。
  • shmflg: 创建共享内存的标志,通常为权限标志(例如 0666),也可以与 IPC_CREATIPC_EXCL 等组合使用。
    • SHM_HUGETLB:类似于mmap的MAP_HUGETLB标志,系统将使用“大页面”来为共享内存分配空间。
    • SHM_NORESERVE,类似于mmap的MAP_NORESERVE标志,不为共享内存保留交换分区(swap空间)。这样,当物理内存不足的时候,对该共享内存执行写操作将触发SIGSEGV信号。
    • IPC_CREAT: 创建新的共享内存,如果创建共享内存, 需要指定对共享内存的操作权限,比如:IPC_CREAT | 0664
    • IPC_EXCL: 检测共享内存是否已经存在了,必须和 IPC_CREAT一起使用

shmget 函数的返回值是共享内存的标识符(如果成功),或者在出错时返回 -1,并设置 errno 来指示错误的原因。

用shmget创建共享内存则这段共享内存的所有字节都被初始化为0,与之关联的内核数据结构shmid_ds将被创建并初始化。shmid_ds结构体的定义如下:

struct shmid_ds {
    struct ipc_perm shm_perm;/*共享内存的操作权限*/
    size_t shm_segsz;        /*共享内存大小,单位是字节*/
    __time_t shm_atime;      /*对这段内存最后一次调用shmat的时间*/
    __time_t shm_dtime;      /*对这段内存最后一次调用shmdt的时间*/
    __time_t shm_ctime;      /*对这段内存最后一次调用shmctl的时间*/
    __pid_t shm_cpid;        /*创建者的PID*/
    __pid_t shm_lpid;        /*最后一次执行shmat或shmdt操作的进程的PID*/
    shmatt_t shm_nattach;    /*目前关联到此共享内存的进程数量,引用计数*/
    /*省略一些填充字段*/
};

1.1、函数使用举例

创建一块大小为4KB的共享内存

shmget(100, 4096, IPC_CREAT|0664);

创建一块大小为4k的共享内存, 并且检测是否存在

// 	如果共享内存已经存在, 共享内存创建失败, 返回-1, 置errno
shmget(100, 4096, IPC_CREAT|0664|IPC_EXCL);

打开一块已经存在的共享内存

// 函数参数虽然指定了大小和IPC_CREAT, 但是都不起作用, 因为共享内存已经存在, 只能打开, 参数4096也没有意义
shmget(100, 4096, IPC_CREAT|0664);
shmget(100, 0, 0);

打开一块共享内存, 如果不存在就创建

shmget(100, 4096, IPC_CREAT|0664);

1.2、ftok

shmget() 函数的第一个参数是一个大于0的正整数,如果不想自己指定可以通过 ftok()函数直接生成这个key值

#include <sys/types.h>
#include <sys/ipc.h>

// 将两个参数作为种子, 生成一个 key_t 类型的数值
key_t ftok(const char *pathname, int proj_id);
  • pathname: 当前操作系统中一个存在的路径
  • proj_id: 这个参数只用到了int中的一个字节, 传参的时候要将其作为 char 进行操作,取值范围: 1-255
  • 返回值:函数调用成功返回一个可用于创建、打开共享内存的key值,调用失败返回-1

二、关联和解除关联

共享内存被创建/获取之后,我们不能立即访问它,而是需要先将它关联到进程的地址空间中。使用完共享内存之后,我们也需要将它从进程地址空间中分离。

#include <sys/shm.h>

void* shmat(int shm_id, const void* shm_addr, int shmflg);
int shmdt(const void* shm_addr);
  • shm_id参数是由shmget调用返回的共享内存标识符

  • shm_addr参数指定将共享内存关联到进程的哪块地址空间,一般来讲用户不知道,需要置NULL

  • shmflg参数的可选标志:

    • 如果shm_addr为NULL,则被关联的地址由操作系统选择
    • 如果shm_addr非空,并且SHM_RND标志未被设置,则共享内存被关联到addr指定的地址处
    • 如果shm_addr非空,并且设置了SHM_RND标志,则被关联的地址是[shm_addr-(shm_addr%SHMLBA)]。SHMLBA的含义是“段低端边界地址倍数”(Segment Low Boundary Address Multiple),它必须是内存页面大小(PAGE_SIZE)的整数倍。现在的Linux内核中,它等于一个内存页大小。这个标志的意义更像是取整。
    • SHM_RDONLY:进程仅能读取共享内存中的内容。若没有指定该标志,则进程可同时对共享内存进行读写操作
    • SHM_REMAP:如果地址shmaddr已经被关联到一段共享内存上,则重新关联。
    • SHM_EXEC:它指定对共享内存段的执行权限。对共享内存而言,执行权限实际上和读权限是一样的。
    • 0:读写权限

三、设置共享内存

3.1、shmctl

shmctl系统调用控制共享内存的某些属性。其定义如下:

#include <sys/shm.h>

int shmctl(int shm_id, int command, struct shmid_ds* buf);
  • shm_id参数是由shmget调用返回的共享内存标识符。

  • command参数指定要执行的命令

命令描述成功返回值
IPC_STAT获取共享内存的当前权限和状态信息0
IPC_SET设置共享内存的权限和状态信息0
IPC_RMID从系统中删除共享内存0
IPC_INFO获取系统共享内存资源配置信息,保存在buf上共享内存数
SHM_LOCK锁定共享内存中的内存页,防止其被交换到磁盘上0
SHM_UNLOCK解锁共享内存中的内存页0
SHM_STAT获取共享内存的当前权限和状态信息(与 IPC_STAT 相同)索引为shm_id的标识符
SHM_INFO获取系统中共享内存的信息共享内存数

3.2、shell

使用ipcs 添加参数-m可以查看系统中共享内存的详细信息

ipcs -m

使用 ipcrm 命令可以标记删除某块共享内存

# key == shmget的第一个参数
$ ipcrm -M shmkey  

# id == shmget的返回值
$ ipcrm -m shmid	

四、共享内存的POSIX方法

利用mmap函数的MAP_ANONYMOUS标志可以实现父、子进程之间的匿名内存共享。通过打开同一个文件,mmap也可以实现无关进程之间的内存共享。Linux提供了另外一种利用mmap在无关进程之间共享内存的方式。这种方式无须任何文件的支持,但它需要先使用如下函数来创建或打开一个POSIX共享内存对象:

#include <sys/mman.h>
#include <sys/stat.h>
#include <fcntl.h>

int shm_open(const char* name, int oflag, mode_t mode);
  • name参数指定要创建/打开的共享内存对象。从可移植性的角度考虑,该参数应该使用“/somename”的格式:以“/”开始,后接多个字符,且这些字符都不是“/”;以“\0”结尾,长度不超过NAME_MAX(通常是255)。

  • oflag参数指定创建方式。它可以是下列标志中的一个或者多个的按位或:

    • O_RDONLY。以只读方式打开共享内存对象
    • O_RDWR。以可读、可写方式打开共享内存对象。
    • O_CREAT。如果共享内存对象不存在,则创建之。此时mode参数的最低9位将指定该共享内存对象的访问权限。共享内存对象被创建的时候,其初始长度为0。
    • O_EXCL。和O_CREAT一起使用,如果由name指定的共享内存对象已经存在,则shm_open调用返回错误,否则就创建一个新的共享内存对象。
    • O_TRUNC。如果共享内存对象已经存在,则把它截断,使其长度为0。
  • shm_open调用成功时返回一个文件描述符。该文件描述符可用于后续的mmap调用,从而将共享内存关联到调用进程。shm_open失败时返回-1,并设置errno。

shm_open创建的共享内存对象使用完之后也需要被删除。这个过程是通过如下函数实现的:

#include <sys/mman.h>
#include <sys/stat.h>
#include <fcntl.h>

int shm_unlink(const char* name);

该函数将name参数指定的共享内存对象标记为等待删除。当所有使用该共享内存对象的进程都使用ummap将它从进程中分离之后,系统将销毁这个共享内存对象所占据的资源。

如果代码中使用了上述POSIX共享内存函数,则编译的时候需要指定链接选项-lrt

五、仿真

写共享内存的进程

int main()
{   
    // 创建共享内存
    int key = ftok("./shm",'A');
    int shm_key = shmget(key,4096,IPC_CREAT|0666);
    // 将共享内存关联到进程的地址空间
    void* shm_addr = shmat(shm_key,NULL,0);
    // 共享内存中写入数据
    const char * str = "你好,我是发送共享内存的线程\n";
    memcpy(shm_addr,str,strlen(str));
    // 休眠十秒
    sleep(10);
    // 解除共享内存与地址空间的关联
    shmctl(shm_key,IPC_RMID,NULL);
    
    return 0;
}

读共享内存的进程

int main()
{
    // 打开共享内存
    int key = ftok("./shm",'A');
    int shm_key = shmget(key,4096,IPC_CREAT|0666);
    // 将共享内存关联到进程的地址空间
    void* shm_addr = shmat(shm_key,NULL,0);
    // 读取共享内存
    printf("共享内存数据: %s\n", (char*)shm_addr);
     // 删除共享内存
    shmctl(shm_key, IPC_RMID, NULL);
    printf("共享内存已经被删除...\n");

    return 0;
}

仿真结果

image-20240423145005136

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

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

相关文章

俊杰测评:电视盒子什么牌子好?电视盒子品牌排行榜

欢迎各位来到俊杰的数码测评频道&#xff0c;每年我会进行数十次电视盒子测评&#xff0c;今年已经买过二十多款电视盒子了&#xff0c;本期的测评主题是电视盒子什么牌子好&#xff0c;通过十天的深入详细对比后我整理了电视盒子品牌排行榜&#xff0c;近期想买电视盒子的可以…

代码随想录算法训练营第五十一天| 309.最佳买卖股票时机含冷冻期,714.买卖股票的最佳时机含手续费,总结

题目与题解 参考资料&#xff1a;买卖股票总结 309.最佳买卖股票时机含冷冻期 题目链接&#xff1a;309.最佳买卖股票时机含冷冻期 代码随想录题解&#xff1a;309.最佳买卖股票时机含冷冻期 视频讲解&#xff1a;动态规划来决定最佳时机&#xff0c;这次有冷冻期&#xff01;|…

python获取文件路径

文件&#xff1a;allpath_parameter.py # 获取当前目录路径 # current_dir os.getcwd() # 获取当前目录路径 realpath00 os.path.abspath(os.path.join(os.path.dirname(os.path.split(os.path.realpath(__file__))[0]), .)) print(realpath00)# 获取当前目录的上级目录路…

Centos安装/更新Docker

首先要配置好Centos 配置好静态IP 替换yum源为阿里云 Docker是什么&#xff1f; Docker 是一种开源的应用容器引擎&#xff0c;让开发者可以打包他们的应用以及依赖包到一个可移植的容器中&#xff0c;然后部署到任何流行的 Linux 机器上。是一种虚拟化的技术&#xff0c;可以把…

linux 编译opencv遇到问题

linux编译opencv4.8的时候遇到问题 Error: suffix or operands invalid for vpand看到很多说法是升级as这个工具的版本&#xff0c;自测是从2.20升级到2.27就可以了

12G-SDI视频分配器JR104D-4K-SDI

JR104D-4K-SDI 12G-SDI分配器1分4,12G-SDI分配器1分2,12G-SDI分配器1分8,机架式12G-SDI分配器1分4&#xff0c;12G-SDI分配器4组1分4&#xff0c;12G-SDI分配器16组1分4&#xff0c; 广播级指标生产厂家。 一、产品介绍&#xff1a; JR104D-4K-SDI视频分配器&#xff0c;是按…

Docker网络模式与cgroup资源控制

前言 在 Docker 中&#xff0c;网络模式和 cgroup 资源控制作为关键功能&#xff0c;对于容器的性能优化和资源管理起着至关重要的作用。本文将介绍 Docker 的网络模式和cgroup资源控制&#xff0c;探讨不同网络模式的特点以及如何利用 cgroup 资源控制机制来有效管理容器的资…

【SSM进阶学习系列丨整合篇】Spring+SpringMVC+MyBatis 框架配置详解

文章目录 一、环境准备1.1、创建数据库和表1.2、导入框架依赖的jar包1.3、修改Maven的编译版本1.4、完善Maven目录1.5、编写项目需要的包1.6、编写实体、Mapper、Service 二、配置MyBatis环境2.1、配置mybatis的主配置文件2.2、编写映射文件2.3、测试环境是否正确 三、配置Spri…

机器学习——过拟合

一、过拟合得表现 模型在训练过程中&#xff0c;除了会出现过拟合现象&#xff0c;还有可能出现欠拟合的情况。相比而言&#xff0c;后者通常发生在建模前期&#xff0c;只要做好特征工程一般可以解决模型欠拟合问题。下图描述了模型在训练数据集上的三种情况&#xff1a; 其…

【深度学习实战(11)】搭建训练框架之dataset,dataloader

一、dataset和dataloader要点说明 在我们搭建自己的网络时&#xff0c;往往需要定义自己的dataset和dataloader&#xff0c;将图像和标签数据送入模型。 &#xff08;1&#xff09;在我们定义dataset时&#xff0c;需要继承torch.utils.data.dataset&#xff0c;再重写三个方法…

文本高效拆分内容,根据空行高效拆分文本内容,文本文档管理更轻松

文本文档是我们日常生活和工作中不可或缺的一部分。然而&#xff0c;随着文本内容的不断增加&#xff0c;如何高效、有序地管理这些文档成为了一个挑战。传统的文本编辑工具往往无法满足我们对于文档整理的需求&#xff0c;而手动整理又费时费力。现在&#xff0c;我们为您带来…

Java实战:确定给定日期是一年的第几天

本次实战&#xff0c;我们将探讨如何确定给定日期是一年中的第几天。为此&#xff0c;我们提供了三种不同的方法&#xff0c;每种方法都有其独特的实现方式和适用场景。 方法一&#xff1a;不使用数组 这种方法通过Scanner类获取用户的输入&#xff0c;包括年份、月份和日期。…

从虚拟化走向云原生,红帽OpenShift“一手托两家”

汽车行业已经迈入“软件定义汽车”的新时代。吉利汽车很清醒地意识到&#xff0c;只有通过云原生技术和数字化转型&#xff0c;才能巩固其作为中国领先汽车制造商的地位。 和很多传统企业一样&#xff0c;吉利汽车在走向云原生的过程中也经历了稳态业务与敏态业务并存带来的前所…

WEB攻防-PHP特性-函数缺陷对比

目录 和 MD5函数 intval ​strpos in_array preg_match str_replace 和 使用 时&#xff0c;如果两个比较的操作数类型不同&#xff0c;PHP 会尝试将它们转换为相同的类型&#xff0c;然后再进行比较。 使用 进行比较时&#xff0c;不仅比较值&#xff0c;还比较变量…

网贷大数据黑名单要多久才能变正常?

网贷大数据黑名单是指个人在网贷平台申请贷款时&#xff0c;因为信用记录较差而被列入黑名单&#xff0c;无法获得贷款或者贷款额度受到限制的情况。网贷大数据黑名单的具体时间因个人信用状况、所属平台政策以及银行审核标准不同而异&#xff0c;一般来说&#xff0c;需要一定…

FebHost:注册国外域名优先考虑可用性还是成本?

在选择域名后缀时&#xff0c;应该优先考虑可用性还是成本&#xff1f;这主要取决于您的具体情况。这两个因素都很重要&#xff0c;您应根据自己的需求进行权衡。 可用性方面&#xff1a;热门的域名后缀&#xff0c;如.com和.net&#xff0c;通常需求量较大&#xff0c;因此可…

数字安全实操AG网址漏洞扫描原理与技术手段分析

在数字化世界的大舞台上&#xff0c;网络安全如同守护者一般&#xff0c;默默保卫着我们的虚拟疆界。当我们在享受互联网带来的便利时&#xff0c;一场无形的战争正在上演。黑客们利用各种手段试图攻破网站的安全防线&#xff0c;而防守方则依靠先进的技术和策略来抵御入侵。其…

安卓studio插件开发(一)本地搭建工程

下载idea 社区版本 建立IDE Plugin工程 点击create就行&#xff0c;新建立的工程长这样 比较重要的文件 build.gradle&#xff1a;配置工程的参数 plugin.xml&#xff1a;设置插件的Action位置 build.gradle.kts内容如下&#xff1a; plugins {id("java")id(&quo…

【VTKExamples::Modelling】第四期 MarchingSquares

很高兴在雪易的CSDN遇见你 VTK技术爱好者 QQ:870202403 公众号:VTK忠粉 前言 本文分享VTK样例MarchingSquares,并解析接口vtkMarchingSquares,希望对各位小伙伴有所帮助! 感谢各位小伙伴的点赞+关注,小易会继续努力分享,一起进步! 你的点赞就是我的动力(^U…

c# winform 控件皮肤

控件皮肤下载&#xff1a; https://download.csdn.net/download/m0_46973223/89225992 步骤&#xff1a; 第一步 将IrisSkin4.dll文件放在debug文件下&#xff0c;选一个或者多个后缀名为.ssk文件&#xff08;各个皮肤文件&#xff09;放在debug文件下。 第二步 解决方案资…