Linux多进程和多线程(六)进程间通信-共享内存
- 多进程(六)
- 共享内存
- 共享内存的创建
- 示例:
- 共享内存删除
- 共享内存映射
- 共享内存映射的创建
- 解除共享内存映射
- 示例:
- 写入和读取共享内存中的数据
- 写入:
- ### 读取:
- 大致操作流程:
多进程(六)
共享内存
- 共享内存是将分配的物理空间直接映射到进程的⽤户虚拟地址空间中, 减少数据在内核空间缓存
- 共享内存是⼀种效率较⾼的进程间通讯的⽅式
在 Linux 系统中通过 ipcs -m 查看所有的共享内存
共享内存的模型:
共享内存的创建
- 使用
shmget()函数创建共享内存
函数头文件:
#include <sys/ipc.h>
#include <sys/shm.h>int shmget (key_t __key, size_t __size, int __shmflg)
key是一个整数值, 用于标识共享内存块, 必须唯一size是一个整数值, 表示共享内存块的大小, 单位为字节shmflg共享内存标志,是一个整数值, 用于设置共享内存的访问权限, 可以取值:IPC_CREAT:创建共享内存块,如果 key 已经存在,则返回错误IPC_EXCL:和 IPC_CREAT 相反,如果 key 已经存在,则返回错误0:访问权限为读写
返回值:
- 如果成功, 则返回一个非负整数, 该整数是共享内存块的标识符
- 如果出错, 则返回 -1, 并设置
errno变量
示例:
#include <sys/ipc.h>
#include <sys/shm.h>
#include <sys/ipc.h>
#include <string.h>
#include <stdio.h>
#include <stdlib.h>
#define SHM_PATH "/home/gopher"
#define SHM_ID 1234
int main() {key_t key;//消息队列的key//通过文件路径和ID生成key,key= ftok(SHM_PATH,SHM_ID);if(key==-1){printf("ftok() error\n");exit(EXIT_FAILURE);}printf("key: %d\n",key);//创建共享内存int shmid;shmid=shmget(key,1024,IPC_CREAT|0666);printf("shmid: %d\n",shmid);return 0;
}
运行结果:
key: -769608541
shmid: 1

共享内存删除
可以通过命令删除共享内存:
ipcrm -m <共享内存标识符>
也可以通过 shmctl() 函数删除共享内存:
函数头文件:
#include <sys/ipc.h>
#include <sys/shm.h>int shmctl (int __shmid, int __cmd, struct shmid_ds *__buf)
函数功能:
共享内存控制函数,功能由具体的功能命令字决定, 用于控制共享内存的创建、删除、设置和获取信息等。
shmid共享内存标识符, 由shmget()函数返回cmd功能命令字, 用于指定控制命令, 可以取值:IPC_RMID:删除共享内存块,使用时候第三个参数为 NULLIPC_SET:设置共享内存块的属性IPC_STAT:获取共享内存块的属性
buf指向shmid_ds结构体(共享内存数据结构)的指针, 用于设置或获取共享内存块的属性- 返回值:
- 如果成功, 则返回 0
- 如果出错, 则返回 -1, 并设置
errno变量
shmid_ds结构体:
struct shmid_ds {struct ipc_perm shm_perm; /* 共享内存权限 */ size_t shm_segsz; /* 共享内存段大小 */pid_t shm_lpid; /* 最后一个 attach 进程的 PID */pid_t shm_cpid; /* 当前 attach 进程的 PID */unsigned long shm_nattch; /* 当前 attach 进程的数量 */time_t shm_atime; /* 上次 attach 时间 */time_t shm_dtime; /* 上次 detach 时间 */time_t shm_ctime; /* 创建时间 */void *shm_internal; /* 内部指针 */
};
删除示例:
#include <sys/ipc.h>
#include <sys/shm.h>
#include <sys/ipc.h>
#include <string.h>
#include <stdio.h>
#include <stdlib.h>
#define SHM_PATH "/home/gopher"
#define SHM_ID 1234
int main() {key_t key;//消息队列的key//通过文件路径和ID生成key,key= ftok(SHM_PATH,SHM_ID);if(key==-1){printf("ftok() error\n");exit(EXIT_FAILURE);}printf("key: %d\n",key);//创建共享内存int shmid;shmid=shmget(key,1024,IPC_CREAT|0666);printf("shmid: %d\n",shmid);int ret=shmctl(shmid,IPC_RMID,NULL); //删除共享内存if(ret==-1){printf("shmctl() error\n");exit(EXIT_FAILURE);}return 0;}
共享内存映射
- 共享内存映射是将共享内存中的数据映射到进程的虚拟地址空间中, 使得进程可以直接访问共享内存中的数据
- 共享内存映射是⼀种效率较⾼的进程间通讯的⽅式
共享内存映射的创建
- 使用
shmat()函数创建共享内存映射 - 函数头文件:
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/shm.h>
void *shmat (int __shmid, const void *__shmaddr, int __shmflg)
函数功能:
将进程地址空间中的一个区域映射到共享内存中, 并返回映射的地址。
shmid共享内存标识符, 由shmget()函数返回shmaddr映射到共享内存中的地址, 可以为NULL, 表示由系统选择映射地址shmflg映射标志, 可以取值:SHM_RDONLY:只读映射SHM_RND:映射地址随机SHM_REMAP:允许映射到已存在的共享内存SHM_EXEC:允许映射到可执行内存- 0 :默认值, 允许读写映射
返回值:
- 如果成功, 则返回映射到共享内存中的地址
- 如果出错, 则返回
(void *)-1, 并设置errno变量
解除共享内存映射
- 使用
shmdt()函数解除共享内存映射 - 函数头文件:
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/shm.h>
int shmdt (const void *__shmaddr)
函数功能:
将进程地址空间中的一个区域与共享内存的映射解除。
-
shmaddr映射到共享内存中的地址 -
返回值:
-
如果成功, 则返回 0
-
如果出错, 则返回 -1, 并设置
errno变量
示例:
#include <sys/ipc.h>
#include <sys/shm.h>
#include <sys/ipc.h>
#include <string.h>
#include <stdio.h>
#include <stdlib.h>
#define SHM_PATH "/home/gopher"
#define SHM_ID 1234
int main() {key_t key;//消息队列的key//通过文件路径和ID生成key,key= ftok(SHM_PATH,SHM_ID);if(key==-1){printf("ftok() error\n");exit(EXIT_FAILURE);}printf("key: %d\n",key);//创建共享内存int shmid;shmid=shmget(key,1024,IPC_CREAT|0666);printf("shmid: %d\n",shmid);// int ret=shmctl(shmid,IPC_RMID,NULL); //删除共享内存
// if(ret==-1){
// printf("shmctl() error\n");
// exit(EXIT_FAILURE);
// }//映射共享内存void* addr=NULL;addr=shmat(shmid,NULL,0);if(addr==(void*)-1){printf("shmat() error\n");exit(EXIT_FAILURE);}printf("addr: %p\n",addr);//修改共享内存中的数据memset(addr,'A',10);//解除映射int ret=shmdt(addr);if(ret==-1) {printf("shmdt() error\n");exit(EXIT_FAILURE);}return 0;
}
写入和读取共享内存中的数据
写入:
#include <sys/ipc.h>
#include <sys/shm.h>
#include <sys/ipc.h>
#include <string.h>
#include <stdio.h>
#include <stdlib.h>
#define SHM_PATH "/home/gopher"
#define SHM_ID 1234
int main() {key_t key;//消息队列的key//通过文件路径和ID生成key,key= ftok(SHM_PATH,SHM_ID);if(key==-1){printf("ftok() error\n");exit(EXIT_FAILURE);}printf("key: %d\n",key);//创建共享内存int shmid;shmid=shmget(key,1024,IPC_CREAT|0666);printf("shmid: %d\n",shmid);// int ret=shmctl(shmid,IPC_RMID,NULL); //删除共享内存
// if(ret==-1){
// printf("shmctl() error\n");
// exit(EXIT_FAILURE);
// }//映射共享内存void* addr=NULL;addr=shmat(shmid,NULL,0);if(addr==(void*)-1){printf("shmat() error\n");exit(EXIT_FAILURE);}printf("addr: %p\n",addr);//修改共享内存中的数据memset(addr,'A',10);//解除映射int ret=shmdt(addr);if(ret==-1) {printf("shmdt() error\n");exit(EXIT_FAILURE);}return 0;
}
读取:
#include <sys/ipc.h>
#include <sys/shm.h>
#include <sys/ipc.h>
#include <string.h>
#include <stdio.h>
#include <stdlib.h>
#define SHM_PATH "/home/gopher"
#define SHM_ID 1234
int main() {key_t key;//消息队列的key//通过文件路径和ID生成key,key= ftok(SHM_PATH,SHM_ID);if(key==-1){printf("ftok() error\n");exit(EXIT_FAILURE);}printf("key: %d\n",key);//创建共享内存int shmid;shmid=shmget(key,1024,IPC_CREAT|0666);printf("shmid: %d\n",shmid);// int ret=shmctl(shmid,IPC_RMID,NULL); //删除共享内存
// if(ret==-1){
// printf("shmctl() error\n");
// exit(EXIT_FAILURE);
// }//映射共享内存void* addr=NULL;addr=shmat(shmid,NULL,0);if(addr==(void*)-1){printf("shmat() error\n");exit(EXIT_FAILURE);}printf("addr: %p\n",addr);//读取共享内存中的数据char str[1024];memcpy(str,addr,1024);printf("str: %s\n",str);printf("strlen(str): %d\n",strlen(str));//解除映射int ret=shmdt(addr);if(ret==-1) {printf("shmdt() error\n");exit(EXIT_FAILURE);}return 0;
}
运行结果
key: -769608541
shmid: 2
addr: 0x7f724c65a000
str: AAAAAAAAAA
strlen(str): 10
大致操作流程:
n",addr);
//读取共享内存中的数据
char str[1024];
memcpy(str,addr,1024);
printf(“str: %s\n”,str);
printf(“strlen(str): %d\n”,strlen(str));
//解除映射
int ret=shmdt(addr);
if(ret==-1) {
printf(“shmdt() error\n”);
exit(EXIT_FAILURE);
}
return 0;
}
运行结果
```c
key: -769608541
shmid: 2
addr: 0x7f724c65a000
str: AAAAAAAAAA
strlen(str): 10
大致操作流程:
相关文章:
Linux多进程和多线程(六)进程间通信-共享内存
多进程(六) 共享内存共享内存的创建 示例: 共享内存删除 共享内存映射 共享内存映射的创建解除共享内存映射示例:写入和读取共享内存中的数据 写入: ### 读取: 大致操作流程: 多进程(六) 共享内存 共享内存是将分配的物理空间直接映射到进程的⽤户虚拟地址空间中, 减少数据在…...
ruoyi后台修改
一、日志文件过大分包 \ruoyi-admin\src\main\resources\logback.xml <!-- 系统日志输出 --> <appender name"file_info" class"ch.qos.logback.core.rolling.RollingFileAppender"><file>${log.path}/sys-info.log</file><!…...
macOS查看系统日志的方法
1、command空格键打开搜索框,输入‘控制台’并打开 2、选择日志报告,根据日期打开自己需要的文件就可以...
数字信号处理及MATLAB仿真(3)——采样与量化
今天写主要来编的程序就是咱们AD变换的两个步骤。一个是采样,还有一个是量化。大家可以先看看,这一过程当中的信号是如何变化的。信号的变换图如下。 先说说采样,采样是将连续时间信号转换为离散时间信号的过程。在采样过程中,连续…...
云端AI大模型群体智慧后台架构思考
1 大模型的调研 1.1 主流的大模型 openai-chatgpt 阿里巴巴-通义千问 一个专门响应人类指令的大模型。我是效率助手,也是点子生成机,我服务于人类,致力于让生活更美好。 百度-文心一言(千帆大模型) 文心一言"…...
算法系列--分治排序|再谈快速排序|快速排序的优化|快速选择算法
前言:本文就前期学习快速排序算法的一些疑惑点进行详细解答,并且给出基础快速排序算法的优化版本 一.再谈快速排序 快速排序算法的核心是分治思想,分治策略分为以下三步: 分解:将原问题分解为若干相似,规模较小的子问题解决:如果子问题规模较小,直接解决;否则递归解决子问题合…...
强化学习编程实战-1-一个及其简单的强化学习实例(多臂赌博机)
1.1 多臂赌博机 一台拥有K个臂的机器,玩家每次可以摇动K个臂中的一个,摇动后,会吐出数量不等的金币,吐出金币的数量服从一定的概率分布,而且不同臂的概率分布不同。 多臂赌博机的问题是:假设玩家共有N次摇地…...
Golang语法规范和风格指南(一)——简单指南
1. 前引 一个语言的规范的学习是重要的,直接关系到你的代码是否易于维护和理解,同时学习好对应的语言规范可以在前期学习阶段有效规避该语言语法和未知编程风格的冲突。 这里是 Google 提供的规范,有助于大家在开始学习阶段对 Golang 进行一…...
数据机构记录顺序表-笔记1
一、线性表的基本概念 数据元素:线性表中的基本单位,每个元素都是线性表的一部分。 数据项:数据元素的具体值。 存储位置:线性表中的元素在内存中的具体存储位置。 线性表按存储结构可以分为顺序表和链表两大类: 1.1…...
考研必备~总结严蔚敏教授《数据结构》课程的重要知识点及考点
作者主页:知孤云出岫 目录 1. 基本概念1.1 数据结构的定义1.2 抽象数据类型 (ADT) 2. 线性表2.1 顺序表2.2 链表 3. 栈和队列3.1 栈3.2 队列 4. 树和二叉树4.1 树的基本概念4.2 二叉树 5. 图5.1 图的基本概念5.2 图的遍历 6. 查找和排序6.1 查找6.2 排序 7. 重点考…...
【数据分享】国家级旅游休闲街区数据(Excel/Shp格式/免费获取)
之前我们分享过从我国文化和旅游部官网整理的2018-2023年我国50个重点旅游城市星级饭店季度经营状况数据(可查看之前的文章获悉详情)!文化和旅游部官网上也分享有很多与旅游相关的常用数据,我们基于官网发布的名单文件整理得到全国…...
Linux开发:进程间通过Unix Domain Socket传递数据
进程间传递数据的方式有很多种,Linux还提供一种特殊的Socket用于在多进程间传递数据,就是Unix Domain Socket(UDS)。 虽然通过普通的Socket也能做到在多进程间传递数据,不过这样需要通过协议栈层的打包与拆包,未免有些浪费效率,通过UDS,数据仅仅通过一个特殊的sock文件…...
Redis基础教程(九):redis有序集合
💝💝💝首先,欢迎各位来到我的博客,很高兴能够在这里和您见面!希望您在这里不仅可以有所收获,同时也能感受到一份轻松欢乐的氛围,祝你生活愉快! 💝Ὁ…...
Servlet与Servlet容器
什么是Servlet? Servlet是Java EE(现称Jakarta EE)中的一个组件,通常用于创建动态Web内容。Servlet是运行在Web服务器上的Java程序,它处理客户端的请求并生成响应。Servlet的核心功能是处理HTTP请求和响应。下面是一个servlet例…...
腾讯centos mysql安装
腾讯centos mysql安装 腾讯云提供了一系列的云计算服务,包括操作系统、数据库、服务器等。在腾讯云上安装CentOS操作系统和MySQL数据库可以按照以下步骤进行: 登录腾讯云控制台(登录 - 腾讯云)。在控制台页面上方的搜索框中输入…...
c_各个unsigned int 和 int的取值范围
bool, uint8_t, uint16_t, uint32_t, uint64_t, int8_t, int16_t, int32_t, int64_t 取值范围分别是什么? 定义形式: typedef unsigned char uint8_t; typedef unsigned short uint16_t; typedef unsigned int uint32_t; typedef unsigned long uint64_…...
C#/WPF 自制截图工具
在日常使用电脑办公时,我们经常遇到需要截图然后保存图片,我们往往需要借助安装截图工具才能实现,现在我们通过C#自制截图工具,也能够轻松进行截图。 我们可以通过C#调用WindousAPI来实现截图,实例代码如下:…...
以腾讯为例,手把手教你搭建产品帮助中心
一个精心设计的产品帮助中心对于提高用户满意度和体验至关重要。腾讯,作为全球领先的互联网企业,通过其多样化的产品线(包括微信、QQ、腾讯游戏、腾讯视频等)吸引了亿万用户。下面将以腾讯为例,向您展示如何搭建一个高…...
计算机网络概述--自我学习用
计算网络体系概述 相关问题 计算机网络为什么要分层?计算机网络是怎么分层的?三种计算机网络模型的关系是什么?每一层分别包含哪些协议?计算机网络中,数据如何在各层中传播?数据在网络各层中的存在形式是…...
超级好用的java http请求工具
kong-http 基于okhttp封装的轻量级http客户端 使用方式 Maven <dependency><groupId>io.github.kongweiguang</groupId><artifactId>kong-http</artifactId><version>0.1</version> </dependency>Gradle implementation …...
基于大模型的 UI 自动化系统
基于大模型的 UI 自动化系统 下面是一个完整的 Python 系统,利用大模型实现智能 UI 自动化,结合计算机视觉和自然语言处理技术,实现"看屏操作"的能力。 系统架构设计 #mermaid-svg-2gn2GRvh5WCP2ktF {font-family:"trebuchet ms",verdana,arial,sans-…...
脑机新手指南(八):OpenBCI_GUI:从环境搭建到数据可视化(下)
一、数据处理与分析实战 (一)实时滤波与参数调整 基础滤波操作 60Hz 工频滤波:勾选界面右侧 “60Hz” 复选框,可有效抑制电网干扰(适用于北美地区,欧洲用户可调整为 50Hz)。 平滑处理&…...
盘古信息PCB行业解决方案:以全域场景重构,激活智造新未来
一、破局:PCB行业的时代之问 在数字经济蓬勃发展的浪潮中,PCB(印制电路板)作为 “电子产品之母”,其重要性愈发凸显。随着 5G、人工智能等新兴技术的加速渗透,PCB行业面临着前所未有的挑战与机遇。产品迭代…...
R语言AI模型部署方案:精准离线运行详解
R语言AI模型部署方案:精准离线运行详解 一、项目概述 本文将构建一个完整的R语言AI部署解决方案,实现鸢尾花分类模型的训练、保存、离线部署和预测功能。核心特点: 100%离线运行能力自包含环境依赖生产级错误处理跨平台兼容性模型版本管理# 文件结构说明 Iris_AI_Deployme…...
Java 8 Stream API 入门到实践详解
一、告别 for 循环! 传统痛点: Java 8 之前,集合操作离不开冗长的 for 循环和匿名类。例如,过滤列表中的偶数: List<Integer> list Arrays.asList(1, 2, 3, 4, 5); List<Integer> evens new ArrayList…...
在HarmonyOS ArkTS ArkUI-X 5.0及以上版本中,手势开发全攻略:
在 HarmonyOS 应用开发中,手势交互是连接用户与设备的核心纽带。ArkTS 框架提供了丰富的手势处理能力,既支持点击、长按、拖拽等基础单一手势的精细控制,也能通过多种绑定策略解决父子组件的手势竞争问题。本文将结合官方开发文档,…...
【网络安全产品大调研系列】2. 体验漏洞扫描
前言 2023 年漏洞扫描服务市场规模预计为 3.06(十亿美元)。漏洞扫描服务市场行业预计将从 2024 年的 3.48(十亿美元)增长到 2032 年的 9.54(十亿美元)。预测期内漏洞扫描服务市场 CAGR(增长率&…...
在Ubuntu中设置开机自动运行(sudo)指令的指南
在Ubuntu系统中,有时需要在系统启动时自动执行某些命令,特别是需要 sudo权限的指令。为了实现这一功能,可以使用多种方法,包括编写Systemd服务、配置 rc.local文件或使用 cron任务计划。本文将详细介绍这些方法,并提供…...
CocosCreator 之 JavaScript/TypeScript和Java的相互交互
引擎版本: 3.8.1 语言: JavaScript/TypeScript、C、Java 环境:Window 参考:Java原生反射机制 您好,我是鹤九日! 回顾 在上篇文章中:CocosCreator Android项目接入UnityAds 广告SDK。 我们简单讲…...
论文浅尝 | 基于判别指令微调生成式大语言模型的知识图谱补全方法(ISWC2024)
笔记整理:刘治强,浙江大学硕士生,研究方向为知识图谱表示学习,大语言模型 论文链接:http://arxiv.org/abs/2407.16127 发表会议:ISWC 2024 1. 动机 传统的知识图谱补全(KGC)模型通过…...

