Linux-共享内存

文章目录

  • 前言
  • 一、system V共享内存
      • 申请共享内存
      • 挂载共享内存
      • 删除共享内存挂载
      • 删除共享内存
  • 二、示例代码
  • 三.运行效果


前言


在这之前我们已经学习了两种进程间通信方式:匿名管道和命名管道。
从我们之前的学习已经知道,想让多个进程间进行通信就需要让他们一起看到同一份资源。
匿名管道是通过fork子进程来让子进程继承父进程的fd。
命名管道是通过生成命名管道文件,并一起打开管道文件。

一、system V共享内存

共享内存相对于我们之前的管道通信有一定区别:

1.共享内存是要让多个进程看到同一份内存.

根据之前我们介绍过的冯洛伊曼体系,对于内存级别的通信特性就代表了共享内存其通信效率要高于管道通信!

2.进程想要看到同一份共享内存,需要key

在这里插入图片描述
这里生成的key方式与哈希字符串类似,通过算法来形成key。所以要想要形成同样的key,就必须确保pathname和porj_id相同,不同进程凭借同样的key来访问同一份共享内存!

申请共享内存

在这里插入图片描述
参数key 代表如果要访问该共享内存需要的key。
参数size代表申请的共享内存大小,这里需要注意的是,共享内存的大小是以4096个字节为单位,所以size最好是4096的倍数。
参数shmflg是模式选项,有 IPC_CREAT 和 IPC_EXCL , IPC_CREAT单独使用代表 如果没有该共享内存则创建,有则使用已经存在的。 IPC_EXCL单独使用没有意义,如果和IPC_CREAT一起使用代表如果没有该共享内存则创建,如果已经存在则报错。
返回值是共享内存的id,就跟文件一样,我们的共享内存也需要进行管理,所以就也有id。

挂载共享内存

由于我们的共享内存的通信方式是让多个进程看到同一份内存,从我们之前学习地址空间的知识,进程需要通过虚拟地址空间->页表->物理内存,所以,要想看到看到位于物理内存的共享内存,就需要修改页表来做到,所以提供了挂载共享内存的接口函数
在这里插入图片描述
参数shmid是我们刚刚讲的共享内存id。
参数shmaddr 可以指定shmaddr的地址为挂载的共享内存地址,一般设置为nullptr。
参数shmflg是模式选项,SHM_RND和SHM_RDONLY,SHM_RND与shmaddr相关,SHM_RDONLY指定该进程只允许对共享内存进行读操作。
返回值为挂载的共享内存地址。

删除共享内存挂载

注意:这里是删除挂载,不是删除共享内存!!!

在这里插入图片描述
参数shmaddr为共享内存在该进程的地址。
返回值若为1则删除成功,-1则发生错误。

删除共享内存

在这里插入图片描述
参数shmid为共享内存id。
参数cmd为模式选项,其中IPC_RMID为删除选项
参数buf这里暂时不讨论。
返回值若为1则删除成功,-1则发生错误

我们要想删除共享内存也不止这一种方式

通过输入ipcs -m 查看存在的共享内存属性
在这里插入图片描述

通过输入ipcrm -m shmid 来删除共享内存

二、示例代码

#Server端
#include "comm.hpp"
#include "Log.hpp"int main()
{// 1.创建创建tokenkey_t key = ftok(PATH_NAME, PROJ_ID);Log(Debug) << "共享秘钥创建成功! step 1"<< " [key:" << getKey(key) << "]" << std::endl;// 2.申请共享内存int shmid = shmget(key, SHM_SIZE, IPC_CREAT | 0666);if (shmid == -1){Log(Error) << "共享内存创建失败!!!!! step 2" << std::endl;perror("shmget");exit(1);}Log(Debug) << "共享内存创建成功! step 2" << std::endl;//sleep(10);char *shmaddr = (char *)shmat(shmid, nullptr, SHM_RDONLY);if ((void *)shmaddr == (void *)-1){Log(Error) << "共享内存挂载失败!!!!!! step 3" << std::endl;perror("shmat");exit(2);}Log(Debug) << "共享内存挂载成功! step 3" << std::endl;// sleep(5);//开始访问共享内存while(1){printf("%s\n",shmaddr);sleep(1);if(strcmp(shmaddr,"quit") == 0) break;}int n = shmdt(shmaddr);if (n == -1){Log(Error) << "共享内存挂载删除失败! step 4" << std::endl;perror("shmdt");exit(3);}Log(Debug) << "共享内存挂载删除! step 4" << std::endl;//sleep(5);n = shmctl(shmid, IPC_RMID, nullptr);if (n == -1){Log(Error) << "共享内存删除失败! step 5" << std::endl;perror("shmctl");exit(4);}Log(Debug) << "共享内存删除成功! step 5" << std::endl;return 0;
}
#Client端
#include "Log.hpp"
#include "comm.hpp"int main()
{key_t key = ftok(PATH_NAME, PROJ_ID);Log(Debug) << "共享秘钥创建成功!step 1"<< " [key:" << getKey(key) << "]" << std::endl;int shmid = shmget(key, SHM_SIZE, 0);if (shmid == -1){Log(Error) << "共享内存获取失败!!!!! step 2" << std::endl;perror("shmget");exit(1);}Log(Debug) << "共享内存获取成功!step 2" << std::endl;//sleep(10);char *shmaddr = (char *)shmat(shmid, nullptr, 0);if ((void *)shmaddr == (void *)-1){Log(Error) << "共享内存挂载失败!!!!!! step 3" << std::endl;   perror("shmat");exit(2);}Log(Debug) << "共享内存挂载成功!step 3" << std::endl;//sleep(5);while(1){//std::cout << "请输入:->" ;ssize_t n = read(0, shmaddr, SHM_SIZE - 1);if(n > 0){shmaddr[n - 1] = 0;if(strcmp(shmaddr,"quit") == 0) break;}}int n = shmdt(shmaddr);if (n == -1){Log(Error) << "共享内存挂载删除失败! step 4" << std::endl;perror("shmdt");exit(3);}Log(Debug) << "共享内存挂载删除!step 4" << std::endl;//sleep(5);return 0;
}
comm.hpp
#include <iostream>
#include <cstdio>
#include <sys/types.h>
#include <sys/ipc.h>
#include <assert.h>
#include <sys/shm.h>
#include <unistd.h>
#include <cstring>
#define PROJ_ID 10086
#define SHM_SIZE 4096char buffer[514] = {0};const char *getKey(key_t key)
{sprintf(buffer, "0x%x", key);return buffer;
}#define PATH_NAME "/home/fengjunzi/test"
Log.hpp
#include <iostream>
#include <time.h>
#include <string>#define Debug 0
#define Error 1const std::string com[] = {"Debug","Error"};std::ostream &Log(int command)
{std::cout << "[" << (unsigned)time(nullptr) << "]:"<< "[" << com[command] << "]" <" ";return std::cout;
}

三.运行效果

在这里插入图片描述
它的缺陷从运行就可以看出来,共享内存没有进行同步与互斥。
不能像管道一样具有访问控制,就会出现写端只写了一半,但是读端已经开始读了的情况。

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

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

相关文章

前端JavaScript篇之intanceof 操作符的实现原理及实现、为什么0.1+0.2 ! == 0.3,如何让其相等

目录 intanceof 操作符的实现原理及实现为什么0.10.2 ! 0.3&#xff0c;如何让其相等 intanceof 操作符的实现原理及实现 instanceof 是一个用于检查对象是否属于特定类的 JavaScript 操作符。它返回一个布尔值&#xff0c;指示对象是否是特定类的实例或者原型链中是否存在该…

常见の算法

前言本文主要使用Java 什么&#xff0c;是快乐星球#&#xffe5;%……什么是算法&#xff1f; 算法是一组完成任务的指令。任何代码片段都可视为算法&#xff0c;但我们主要介绍常见算法 一、引入——二分查找 二分查找是一种算法&#xff0c;其输入是一个有序的元素列表。如…

浏览器实用:禁止浏览器http自动转成https

前言 因为有些网站支持http和https两种协议访问&#xff0c;有些只支持一种协议访问。但根据动态域名安全策略&#xff08;HSTS&#xff09;&#xff0c;只要该域名在浏览器中访问过一次https&#xff0c;那么浏览器将强制使http的请求变为https。 虽然这条浏览器的策略有利于提…

Flink入门教程

使用flink时需要提前准备好scala环境 一、创建maven项目 二、添加pom依赖 <properties><scala.version>2.11.12</scala.version></properties><dependency><groupId>org.scala-lang</groupId><artifactId>scala-library<…

携程基于Jira Cloud的敏捷项目管理实践

好的工具可以满足团队在各个成长阶段的管理诉求 实践一&#xff1a;对齐目标/团队OKR/多团队协作战略项目 实践二&#xff1a;以产品为中心的协作框架 实践三&#xff1a;交付团队管理 实践四&#xff1a;和海外子公司对齐&#xff0c;协作

在linux、window环境搭建kafka环境

一、搭建环境前置准备 下载kafka的官网 http://kafka.apache.org/downloads根据自己的需求选择版本,安装包不区分linux和windows环境,这一个安装包均可部署。 源代码包含kafka的代码文件,使用scala编写的。 二、linux环境 1. 上传安装包 我下载的版本是kafka_2.12-3.6.1…

09.Elasticsearch应用(九)

Elasticsearch应用&#xff08;九&#xff09; 1.搜索结果处理包括什么 排序分页高亮返回指定字段 2.排序 介绍 Elasticsearch支持对搜索结果排序&#xff0c;默认是根据相关度算分来排序 支持排序的字段 keyword数值地理坐标日期类型 排序语法 GET /[索引名称]/_sear…

网络协议与攻击模拟_08DHCP协议

技术学习要了解某项技术能干什么&#xff1f;它的详细内容&#xff1f;发展走向&#xff1f; 一、DHCP协议 1、DHCP基本概念 dhcp动态主机配置协议&#xff0c;广泛应用于局域网内部 主要是为客户机提供TCP/IP 参数&#xff08;IP地址、子网掩码、网关、DNS等&#xff09;…

5G赋能智慧文旅:科技与文化的完美结合,打造无缝旅游体验,重塑旅游业的未来

一、5G技术&#xff1a;智慧文旅的强大引擎 5G技术的起源可以追溯到2010年&#xff0c;当时世界各国开始意识到4G技术已经达到了瓶颈&#xff0c;无法满足日益增长的移动通信需求。2013年&#xff0c;国际电信联盟&#xff08;ITU&#xff09;成立了5G技术研究组&#xff0c;开…

超融合系统疑难故障定位与解决实践 3 例(含信创技术栈)

当 IT 系统出现故障&#xff0c;问题定位往往是运维人员最头疼的环节。尤其是超融合系统&#xff0c;由于整体涉及的技术栈比较复杂&#xff0c;且有越来越多的用户基于信创环境进行部署&#xff0c;非常考验厂商和技术人员的专业能力&#xff1a;厂商研发和售后工程师不仅应能…

[pytorch入门] 6. 神经网络

基本介绍 torch.nn&#xff1a; Containers&#xff1a;基本骨架Convolution Layers&#xff1a; 卷积层Pooling layers&#xff1a;池化层Non-linear Activations (weighted sum, nonlinearity)&#xff1a;非线性激活Normalization Layers&#xff1a;正则化层 Container…

一文讲透Excel数据如何导入到Stata?

推荐采用《Stata统计分析从入门到精通》 杨维忠、张甜 清华大学出版社“1.2.6 导入其他格式的数据文件” 的解答。 在Stata主界面选择“文件|导入”命令&#xff08;如图所示&#xff09;&#xff0c;即可看到Stata支持的其他格式的数据文件类型&#xff0c;包括Excel电子表格…

【收藏】幻兽帕鲁创建服务器教程

创建幻兽帕鲁服务器1分钟部署教程&#xff0c;阿里云和腾讯云均推出幻兽帕鲁服务器服务器和部署教程&#xff0c;4核16G和4核32G配置可选&#xff0c;阿腾云atengyun.com分享1分钟自建幻兽帕鲁Palworld服务器教程&#xff1a; 幻兽帕鲁服务器创建教程 幻兽帕鲁服务器官方推荐…

如何在 Linux 上安装 Docker?最权威指南来了!

docker文档&#xff1a;https://docs.docker.com/engine/ 打开docker文档 这里需要注意&#xff0c;服务器上一般是没有界面的&#xff0c;我们只需要安装docker Engine。 卸载历史版本 这一步是可选的&#xff0c;如果之前安装过旧版本的Docker&#xff0c;可以使用如下命令…

[RK-Linux] 移植Linux-5.10到RK3399(十)| 配置AP6256模组使能WIFI、BT功能

手上 ROC-RK3399-PC Pro 使用蓝牙 WIFI 模组是 AP6256。 一、AP6256 模组介绍 AP6256是正基科技(AMPAK)推出的一款低成本、低功耗的双模模块,它集成了Wi-Fi和蓝牙功能。这款模块支持SDIO接口,具有以下特点: 1、型号:AP6256 2、接口:SDIO(Secure Digital Input/Outp…

图像旋转角度计算并旋转

#!/usr/bin/python3 # -*- coding: utf-8 -*- import cv2 import numpy as np import timedef Rotate(img, angle0.0,fill0):"""旋转:param img:待旋转图像:param angle: 旋转角度:param fill&#xff1a;填充方式&#xff0c;默认0黑色填充:return: img: 旋转后…

《幻兽帕鲁》32人专有服务器设置教程,亲测稳定

创建幻兽帕鲁服务器1分钟部署教程&#xff0c;阿里云和腾讯云均推出幻兽帕鲁服务器服务器和部署教程&#xff0c;4核16G和4核32G配置可选&#xff0c;阿腾云atengyun.com分享1分钟自建幻兽帕鲁Palworld服务器教程&#xff1a; 幻兽帕鲁服务器创建教程 幻兽帕鲁服务器官方推荐…

飞速(FS)400G产品全家福及其应用介绍

随着大型数据中心不断向更高性能、更大规模的架构演进&#xff0c;构建能够有效应对海量用户群体、智能设备激增及复杂应用负载所需的高容量网络连接已变得不可或缺。尤其是在超大规模云计算服务迅速普及的背景下&#xff0c;对具备超高带宽和微秒级低延迟特性的网络基础设施需…

灌区信息化系统的建设内容和应用

一、背景 随着科技的不断发展&#xff0c;信息化技术已经逐渐渗透到各个领域中&#xff0c;为我们的生活和工作带来了极大的便利。灌区作为农业发展的重要组成部分&#xff0c;其信息化系统的建设也日益受到重视。 二、政策 据水利部消息&#xff0c;水利部、国家发改委近日正…

Unity中URP下获取额外灯数量

文章目录 前言一、SimpleLit下额外灯数量的获取1、在 SimpleLit 下&#xff0c;先获取了额外灯的数量2、对其进行循环计算每一个额外灯3、GetAdditionalLightsCount在这里插入图片描述 二、GetAdditionalLightsCount实现了什么1、_AdditionalLightsCount.x2、unity_LightData.y…