Linux操作系统4-进程间通信5(共享内存实现两个进程通信)
上篇文章:Linux操作系统4-进程间通信4(共享内存原理,创建,查看,命令)-CSDN博客
本篇Gitee仓库:myLerningCode/l24 · 橘子真甜/Linux操作系统与网络编程学习 - 码云 - 开源中国 (gitee.com)
本篇重点:进程关联与去关联共享内存,实现两个进程通信
一. 进程关联共享内存
两个进程想要通过共享内存进行通信,在创建共享内存之后还需要将这个共享内存与我们的进程相关联。

进程关联共享内存的系统调用如下:
//头文件
#include <sys/types.h>
#include <sys/shm.h>//函数原型
void* shmat(int shmid, const void* shmarr, int shmflg)//参数
shimid 想要关联共享内存的id
shmarr 想要关联共享内存的地址空间,一般设置为NULL
shmflg 一般设置为0//返回值
返回共享内存通过页表映射到进程地址空间的起始地址
二. 进程去关联共享内存
在我们使用完了这段共享内存后,如果想要让其他进程进行通信。可以让当前的进程与共享内存去关联
系统调用如下:
//头文件
#include <sys/types.h>
#include <sys/shm.h>int shmdt(const void* shmaddr);shmaddr 我们关联共享内存时候获取的地址返回值:成功返回0,失败返回-1
三. 删除共享内存
在上篇文章中,我们使用命令 ipcs -m 查看共享内存。使用命令 ipcrm -m 删除共享内存
我们也可以通过系统调用来删除共享内存
系统调用如下:
#include <sys/ipc.h>
#include <sys/shm.h>//用于对共享内存的控制(包含删除)
int shmctl(int shmid, int cmd, struct shmid_ds *buf)cmd:传入的二进制标志位
IPC_RMID:立即移除对应的共享内存buf:删除的话直接设置为NULL即可返回值,失败返回-1,错误码被设置
四. 实现通信
4.1 common.hpp
这个头文件用于实现创建,关联,去关联,删除共享内存的代码。之后只需要让server.cpp和client.cpp去调用代码即可。
创建共享内存:对于client和server来说,需要由不同的操作,server需要创建共享内存,并且存在共享内存需要出错返回,而client只需要获取共享内存的id即可
代码如下:
#include <iostream>#include <cassert>
#include <cstring>
#include <cstdlib>#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/shm.h>
#include <error.h>
#include <unistd.h>// 当前的绝对路径用于获取key
#define PATHNAME "."
#define PROJ_ID 2025221
#define MAXSIZE 5// 获取key值
key_t getKey()
{// 获取一个keykey_t key = ftok(PATHNAME, PROJ_ID);assert(key != -1);
}// 传入key,创建共享内存,并返回其id
// flag是为了区分不同的进程,创建共享内存的进程需要出错返回,而另一个进程则不需要
int getShmHelp(const key_t key, int flag)
{int sid = shmget(key, MAXSIZE, flag);assert(sid >= 0);
}// 共享内存存在,需要出错返回,并且提供权限
int creatShm(key_t key)
{return getShmHelp(key, IPC_CREAT | IPC_EXCL | 0666);
}// 存在共享内存,不做任何事,只是返回id
int getShm(key_t key)
{return getShmHelp(key, IPC_CREAT);
}
关联共享内存:传入id,关联共享内存,返回地址空间的起始地址即可
// 关联共享内存与进程
void *attachShm(int shmid)
{void *mem = shmat(shmid, nullptr, 0);if ((long long)mem == -1l) // 指针占8字节{std::cerr << "attachShm error" << errno << ":" << strerror(errno) << std::endl;exit(1);}return mem;
}
去关联共享内存:传入id,去关联即可
// 去关联共享内存
void detachShm(void *start)
{int n = shmdt(start);if (n == -1){std::cerr << "detachShm error" << errno << ":" << strerror(errno) << std::endl;exit(2);}
}
删除共享内存:传入id,删除即可
// 删除共享内存
void deleteShm(int shmid)
{int n = shmctl(shmid, IPC_RMID, nullptr);if (n == -1){std::cerr << "deleteShm error" << errno << ":" << strerror(errno) << std::endl;exit(3);}
}
4.2 server.cpp
共享内存通信让进程地址空间与共享内存通过页表进行了映射,所以我们只需要直接向字符串一样输出信息即可。
server端接收来自client端发送的消息
代码如下:
#include "common.hpp"
#include <iostream>
#include <string>int main()
{// 1.创建共享内存int shmid = creatShm(getKey());std::cout << "server 创建共享内存成功" << std::endl;// 2.关联共享内存和进程地址空间的映射char *start = (char *)attachShm(shmid);std::cout << "server 关联共享内存成功" << std::endl;sleep(5);// 开始通信while (true){// 服务端接收信息,由于有地址空间的映射。// 我们只需要和字符串一样接收即可std::cout << start << std::endl;sleep(1);}// 3.去关联共享内存与进程地址空间的映射detachShm(start);std::cout << "server 去关联共享内存成功" << std::endl;// 4.强制删除共享内存deleteShm(shmid);std::cout << "server 删除共享内存成功" << std::endl;return 0;
}
4.3 client.cpp
client端无需删除共享内存
#include "common.hpp"
#include <iostream>int main()
{// 获取共享内存idint shmid = getShm(getKey());std::cout << "client 获取共享内存id成功" << " sid为" << shmid << std::endl;// 关联共享内存char *start = (char *)attachShm(shmid);std::cout << "client 关联共享内存成功" << std::endl;// 开始通信const char *message = "你好server,我是client ...";int cnt = 0;while (true){snprintf(start, MAXSZIE, "%s[client pid:%d][消息编号:%d]", start, getpid(), cnt++);sleep(1);}// 去关联共享内存detachShm(start);std::cout << "client 去关联共享内存成功" << std::endl;// client无需删除共享内存,server删除return 0;
}
五. 分析测试结果
运行测试结果如下:
先运行server,在运行ckient。可以看到server端成功接收来自client的消息

我们让client关闭

关闭client端之后,server端仍然打印最后一条65编号的信息。
这说明共享内存没有同步和互斥功能,如果想要实现这个功能需要我们加锁
六. 共享内存的特定与优缺点
6.1 优点
共享内存的优点就是
1 是所有进程间通信中最快的,只要我们一方写入数据,另外一方可以直接看到,大大减少数据的拷贝次数
因为共享内存不需要缓冲区,所以速度快。这里我们对比共享内存与管道通信
管道通信:

共享内存通信
在不考虑键盘输入,显示器输出的情况下。
管道文件至少需要拷贝数据4次,而共享内存只需要拷贝数据两次。
很明显。共享内存大大减少的数据的拷贝次数
6.2 缺点
在上面的测试中我们知道,共享内存的缺点是:不支持同步和互斥功能,如果读取速度较快,会读取到已经读取了的数据
相关文章:
Linux操作系统4-进程间通信5(共享内存实现两个进程通信)
上篇文章:Linux操作系统4-进程间通信4(共享内存原理,创建,查看,命令)-CSDN博客 本篇Gitee仓库:myLerningCode/l24 橘子真甜/Linux操作系统与网络编程学习 - 码云 - 开源中国 (gitee.com) 本篇重…...
sam2 windows 编译安装
目录 1. pip install sam2 2. 编译安装 1. pip install sam2 运行报错: cannot import name _C from sam2 (E:\project\smpl\render_blender\linux\GroundedSAM2_SMPL\sam2\__init__.py) 2. 编译安装 cd E:\project\sam2\sam2-main set DISTUTILS_USE_SDK1 py…...
RFID测温技术:电力设备安全监测的新利器
在当今高度依赖电力的现代化社会中,稳定且可靠的电力供应是社会运转的基石。电力设备作为电力系统的关键核心,其运行状态直接关乎电力供应的品质。然而,电力设备长期运行过程中,受到诸如过载、接触不良以及环境因素等多重影响&…...
(一)趣学设计模式 之 单例模式!
目录 一、啥是单例模式?二、为什么要用单例模式?三、单例模式怎么实现?1. 饿汉式:先下手为强! 😈2. 懒汉式:用的时候再创建! 😴3. 枚举:最简单最安全的单例&a…...
自动化办公|xlwings生成图表
在日常的数据分析和报告生成中,Excel图表是一个非常重要的工具。它能够帮助我们直观地展示数据,发现数据中的规律和趋势。然而,手动创建和调整图表往往耗时且容易出错。幸运的是,借助Python的xlwings库,我们可以自动化…...
Docker基于Ollama本地部署大语言模型
一、Ollama介绍 Ollama 是一个开源的大型语言模型(LLM)平台,旨在简化大型语言模型在本地环境中的运行、管理和交互。通过Ollama,用户可以轻松加载和使用各种预训练的语言模型,执行诸如文本生成、翻译、代码编写、问答…...
Pytorch实现之GIEGAN(生成器信息增强GAN)训练自己的数据集
简介 简介:在训练数据样本之前首先利用VAE来推断潜在空间中不同类的分布,用于后续的训练,并使用它来初始化GAN。与ACGAN和BAGAN不同的是,提出的GIEGAN有一个分类器结构,这个分类器主要判断生成的图像或者样本图像属于哪个类,而鉴别器仅判断图像是来自于生成器还是真实样…...
centos9安装k8s集群
以下是基于CentOS Stream 9的Kubernetes 1.28.2完整安装流程(containerd版): 一、系统初始化(所有节点执行) # 关闭防火墙 systemctl disable --now firewalld# 关闭SELinux sed -i "s/SELINUXenforcing/SELINU…...
pytest下allure
import pytestdef test_case01():用例01~print(用例01)class Test_mokuai01:def test_case02(self):用例02~print(用例02)if __name____main__:#pytest.main([-vs,test_sample-2.py])pytest.main([-vs,test_sample-2.py,--allure-dir,./result2])#生成allure报告,参…...
JVM预热
阿里电商平台每年的各种大促活动,对于Java技术来说,其中重要一个操作环节就是预热操作。 目录 预热是什么?为什么要预热? java 程序不预热和预热的调用对比 预热是什么? 预热是指,在 JVM 启动后࿰…...
【JavaWeb12】数据交换与异步请求:JSON与Ajax的绝妙搭配是否塑造了Web的交互革命?
文章目录 🌍一. 数据交换--JSON❄️1. JSON介绍❄️2. JSON 快速入门❄️3. JSON 对象和字符串对象转换❄️4. JSON 在 java 中使用❄️5. 代码演示 🌍二. 异步请求--Ajax❄️1. 基本介绍❄️2. JavaScript 原生 Ajax 请求❄️3. JQuery 的 Ajax 请求 &a…...
Ubuntu 查看mysql用户和数据库
在Ubuntu系统中,你可以使用以下MySQL命令来查看用户和数据库的信息。请确保你已经安装了MySQL服务器,并且你具有足够的权限(如root用户权限)来执行这些命令。 查看所有数据库 要查看MySQL服务器上的所有数据库,可以使…...
数据库服务器和应用服务器的区别是什么?
在当今的互联网社会中,各个行业中的业务都离不来网络科技,而作为互联网基础设施的服务器,受到了各大行业的广泛应用,同时根据所承担职责的不同,可以将服务器分为数据库服务器和应用服务器,本文就来概括一下…...
初级银行从业考试真题
2023 年 6 月初级银行从业考试真题 法律法规 单选题 1.按照《中华人民共和国反洗钱法》的规定,金融机构所建立的客户身份资料和客户交易信息在业务关系或交易结束后至少 保存期限为()年。 A.5 B.3 C.10 D.2 参考答案:A 2.物价稳定是要保持()的大体稳定,避免出现高…...
普通转录组RNASeq生物信息流程
探序基因肿瘤研究院 整理 比对工具:HISAT2,下载源代码编译安装或者二进制文件 定量工具:feactureCounts,下载地址:The Subread package 参考基因组:NCBI的GCF_000001405.40_GRCh38.p14_genomic.fna.g…...
nginx容器配置fastapi服务失败
问题描述: Linux虚拟机中启动了一个fastapi服务器(8000端口),希望能通过nginx容器设置代理使得前端代码可以调用这个接口,但是访问时报错(状态码:502)。nginx配置如下: l…...
网页制作06-html,css,javascript初认识のhtml如何建立超链接
超链接有外部链接、电子邮件链接、锚点链接、空链接、脚本链接 一、内部链接 与自身网站页面有关的链接被称为内部链接 1、创建内部链接 1)语法: <a href"链接地址"> …… </a> 2)举例应用: 3…...
代码讲解系列-CV(七)——前沿论文复现
文章目录 一、论文速览1.1 确定baseline1.2 DepthMaster: Taming Diffusion Models for Monocular Depth Estimation 二、数据环境搭建2.1 环境搭建2.2 数据权重 三、推理debug3.1 单图推理3.2 数据集验证 四、模型训练4.1 数据读取4.2 训练流程 五、作业 一、论文速览 1.1 确…...
3DGS(三维高斯散射)与SLAM技术结合的应用
3DGS(三维高斯散射)与SLAM(即时定位与地图构建)技术的结合,为动态环境感知、高效场景建模与实时渲染提供了新的可能性。以下从技术融合原理、应用场景、优势挑战及典型案例展开分析: 一、核心融合原理 1. …...
数据库面试知识点总结
目录 1. MySQL 基础题1.1 执行⼀条 select / update 语句,在 MySQL 中发生了什么?1.2 MySQL 一行记录是怎么存储的? 2. 三大范式3. 数据库引擎3.1 Innodb3.2 MyISAM 4. 数据库索引4.1 索引分类4.2 索引优缺点4.3 索引使用场景4.4 优化索引方法…...
1.25作业
1easytornado SSTI——tornado模板 hints.txt:在/fllllllllllllag里;计算filehash的方法(需要cookie_secret,对filename进行md5拼接再第二次md5) ?filename/hints.txt&filehash{ {2*3}},跳转到另一个页面 存在且…...
Power Query M函数
文章目录 三、PQ高阶技能:M函数3.1 M函数基本概念3.1.1 表达式和值3.1.2 计算3.1.3 运算符3.1.4 函数3.1.5 元数据3.1.6 Let 表达式3.1.6 If 表达式3.1.7 Error 3.2 自定义M函数3.2.1 语法3.2.2 调用定义好的自定义函数3.2.3 直接调用自定义函数3.2.4 自定义函数&am…...
python argparse 解析命令行参数
可选参数 带 - 或者 -- 的参数都是可选参数,如果命令行不输入,得到的结果是 None 参数名只能使用下划线,不能使用中划线 default: 设置默认值 action: 默认是 store 方法,常用的是 store_true 命令行出…...
使用西门子 PLC(以 S7 - 1200 为例)编写梯形图程序来根据转速计算瞬时流量和累计流量的详细步骤
以下是一个使用西门子 PLC(以 S7 - 1200 为例)编写梯形图程序来根据转速计算瞬时流量和累计流量的详细步骤,同时会考虑与昆仑通泰触摸屏的交互。该程序支持 4 - 20 毫安信号输入和另一种模拟的手动输入方式。 需求理解 流量计算原理&#x…...
【网络编程】服务器模型(二):并发服务器模型(多线程)和 I/O 复用服务器(select / epoll)
一、多线程并发服务器 在 高并发的 TCP 服务器 中,单线程或 fork() 多进程 方式会导致 资源浪费和性能瓶颈。因此,我们可以使用 多线程 来高效处理多个客户端的连接。 承接上文中的多进程并发服务器,代码优化目标: 1.使用 pthr…...
一文读懂大模型文件后缀名,解锁 AI 世界的密码
在大模型的世界里,各种文件后缀名就像一把把钥匙,打开通往不同应用和功能的大门。今天,咱们就来聊聊那些常见又重要的大模型文件后缀名。 safetensors:安全与高效的守护者 safetensors 是一种基于 Python 的序列化格式ÿ…...
探索 Peewee:轻量级 Python ORM 简明指南
文章目录 探索 Peewee:轻量级 Python ORM 简明指南主要特点:安装:使用示例:1. 定义模型:2. 初始化数据库:3. 数据操作(增、查、改、删):4. 查询构建器:5. 迁移…...
生产者标签(Producer Tag) 和 普通连接(显式或隐式连接)的区别:
在 EtherNet/IP 的通信架构中,生产者标签(Producer Tag) 和 普通连接(显式或隐式连接)的区别主要体现在 通信模式、实时性、数据流向 和 资源管理 上。以下是详细对比: 1. 定义与用途 类型生产者标签&…...
【速写】解码与kv-cache的简单细节
问题:在使用transformers调用huggingface生成式模型时,我们可以通过model.generate_ids来获取模型生成的若干id,那么我应该如何查看这些生成id对应的logits大小(即生成概率) 在Hugging Face的Transformers库中&#x…...
Mac【卸载 Python】 - 3.12.2
一、若使用官方安装包安装 1. 删除 Python 框架 Python 官方安装包会将 Python 安装到 /Library/Frameworks/Python.framework/Versions/3.12 目录下。你可以在终端中使用以下命令删除该目录: sudo rm -rf /Library/Frameworks/Python.framework/Versions/3.12 …...
