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 …...
内存分配函数malloc kmalloc vmalloc
内存分配函数malloc kmalloc vmalloc malloc实现步骤: 1)请求大小调整:首先,malloc 需要调整用户请求的大小,以适应内部数据结构(例如,可能需要存储额外的元数据)。通常,这包括对齐调整,确保分配的内存地址满足特定硬件要求(如对齐到8字节或16字节边界)。 2)空闲…...
可靠性+灵活性:电力载波技术在楼宇自控中的核心价值
可靠性灵活性:电力载波技术在楼宇自控中的核心价值 在智能楼宇的自动化控制中,电力载波技术(PLC)凭借其独特的优势,正成为构建高效、稳定、灵活系统的核心解决方案。它利用现有电力线路传输数据,无需额外布…...
java调用dll出现unsatisfiedLinkError以及JNA和JNI的区别
UnsatisfiedLinkError 在对接硬件设备中,我们会遇到使用 java 调用 dll文件 的情况,此时大概率出现UnsatisfiedLinkError链接错误,原因可能有如下几种 类名错误包名错误方法名参数错误使用 JNI 协议调用,结果 dll 未实现 JNI 协…...
Vue2 第一节_Vue2上手_插值表达式{{}}_访问数据和修改数据_Vue开发者工具
文章目录 1.Vue2上手-如何创建一个Vue实例,进行初始化渲染2. 插值表达式{{}}3. 访问数据和修改数据4. vue响应式5. Vue开发者工具--方便调试 1.Vue2上手-如何创建一个Vue实例,进行初始化渲染 准备容器引包创建Vue实例 new Vue()指定配置项 ->渲染数据 准备一个容器,例如: …...
让AI看见世界:MCP协议与服务器的工作原理
让AI看见世界:MCP协议与服务器的工作原理 MCP(Model Context Protocol)是一种创新的通信协议,旨在让大型语言模型能够安全、高效地与外部资源进行交互。在AI技术快速发展的今天,MCP正成为连接AI与现实世界的重要桥梁。…...
Map相关知识
数据结构 二叉树 二叉树,顾名思义,每个节点最多有两个“叉”,也就是两个子节点,分别是左子 节点和右子节点。不过,二叉树并不要求每个节点都有两个子节点,有的节点只 有左子节点,有的节点只有…...
免费数学几何作图web平台
光锐软件免费数学工具,maths,数学制图,数学作图,几何作图,几何,AR开发,AR教育,增强现实,软件公司,XR,MR,VR,虚拟仿真,虚拟现实,混合现实,教育科技产品,职业模拟培训,高保真VR场景,结构互动课件,元宇宙http://xaglare.c…...
Caliper 负载(Workload)详细解析
Caliper 负载(Workload)详细解析 负载(Workload)是 Caliper 性能测试的核心部分,它定义了测试期间要执行的具体合约调用行为和交易模式。下面我将全面深入地讲解负载的各个方面。 一、负载模块基本结构 一个典型的负载模块(如 workload.js)包含以下基本结构: use strict;/…...
Python 实现 Web 静态服务器(HTTP 协议)
目录 一、在本地启动 HTTP 服务器1. Windows 下安装 node.js1)下载安装包2)配置环境变量3)安装镜像4)node.js 的常用命令 2. 安装 http-server 服务3. 使用 http-server 开启服务1)使用 http-server2)详解 …...
从面试角度回答Android中ContentProvider启动原理
Android中ContentProvider原理的面试角度解析,分为已启动和未启动两种场景: 一、ContentProvider已启动的情况 1. 核心流程 触发条件:当其他组件(如Activity、Service)通过ContentR…...

