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 …...
相机Camera日志实例分析之二:相机Camx【专业模式开启直方图拍照】单帧流程日志详解
【关注我,后续持续新增专题博文,谢谢!!!】 上一篇我们讲了: 这一篇我们开始讲: 目录 一、场景操作步骤 二、日志基础关键字分级如下 三、场景日志如下: 一、场景操作步骤 操作步…...
Day131 | 灵神 | 回溯算法 | 子集型 子集
Day131 | 灵神 | 回溯算法 | 子集型 子集 78.子集 78. 子集 - 力扣(LeetCode) 思路: 笔者写过很多次这道题了,不想写题解了,大家看灵神讲解吧 回溯算法套路①子集型回溯【基础算法精讲 14】_哔哩哔哩_bilibili 完…...
java调用dll出现unsatisfiedLinkError以及JNA和JNI的区别
UnsatisfiedLinkError 在对接硬件设备中,我们会遇到使用 java 调用 dll文件 的情况,此时大概率出现UnsatisfiedLinkError链接错误,原因可能有如下几种 类名错误包名错误方法名参数错误使用 JNI 协议调用,结果 dll 未实现 JNI 协…...
江苏艾立泰跨国资源接力:废料变黄金的绿色供应链革命
在华东塑料包装行业面临限塑令深度调整的背景下,江苏艾立泰以一场跨国资源接力的创新实践,重新定义了绿色供应链的边界。 跨国回收网络:废料变黄金的全球棋局 艾立泰在欧洲、东南亚建立再生塑料回收点,将海外废弃包装箱通过标准…...
拉力测试cuda pytorch 把 4070显卡拉满
import torch import timedef stress_test_gpu(matrix_size16384, duration300):"""对GPU进行压力测试,通过持续的矩阵乘法来最大化GPU利用率参数:matrix_size: 矩阵维度大小,增大可提高计算复杂度duration: 测试持续时间(秒&…...
【OSG学习笔记】Day 16: 骨骼动画与蒙皮(osgAnimation)
骨骼动画基础 骨骼动画是 3D 计算机图形中常用的技术,它通过以下两个主要组件实现角色动画。 骨骼系统 (Skeleton):由层级结构的骨头组成,类似于人体骨骼蒙皮 (Mesh Skinning):将模型网格顶点绑定到骨骼上,使骨骼移动…...
Maven 概述、安装、配置、仓库、私服详解
目录 1、Maven 概述 1.1 Maven 的定义 1.2 Maven 解决的问题 1.3 Maven 的核心特性与优势 2、Maven 安装 2.1 下载 Maven 2.2 安装配置 Maven 2.3 测试安装 2.4 修改 Maven 本地仓库的默认路径 3、Maven 配置 3.1 配置本地仓库 3.2 配置 JDK 3.3 IDEA 配置本地 Ma…...
10-Oracle 23 ai Vector Search 概述和参数
一、Oracle AI Vector Search 概述 企业和个人都在尝试各种AI,使用客户端或是内部自己搭建集成大模型的终端,加速与大型语言模型(LLM)的结合,同时使用检索增强生成(Retrieval Augmented Generation &#…...
【笔记】WSL 中 Rust 安装与测试完整记录
#工作记录 WSL 中 Rust 安装与测试完整记录 1. 运行环境 系统:Ubuntu 24.04 LTS (WSL2)架构:x86_64 (GNU/Linux)Rust 版本:rustc 1.87.0 (2025-05-09)Cargo 版本:cargo 1.87.0 (2025-05-06) 2. 安装 Rust 2.1 使用 Rust 官方安…...
快刀集(1): 一刀斩断视频片头广告
一刀流:用一个简单脚本,秒杀视频片头广告,还你清爽观影体验。 1. 引子 作为一个爱生活、爱学习、爱收藏高清资源的老码农,平时写代码之余看看电影、补补片,是再正常不过的事。 电影嘛,要沉浸,…...

