【Linux】共享内存

文章目录

  • 一、共享内存的原理
  • 详谈共享内存的实现过程
    • 二、共享内存的接口函数
      • 1.shmget
      • 2. shmat
      • shmdt
      • shmctl
    • 进程间使用共享内存通信
    • 三、共享内存的特性
  • 关于代码


一、共享内存的原理

共享内存是由操作系统维护和管理的一块内存。

共享内存的本质是内核级的缓冲区。

一个进程向操作系统申请一块共享区内存,操作系统为该进程创建了一块内存后,进程要将该共享内存与自己的虚拟地址空间进行映射挂接。

也就是将共享区内存通过页表建立映射关系后,在进程自己的虚拟地址空间的共享区中就保留了共享内存的起始地址。

在这里插入图片描述

同时,进程b也通过页表映射,将共享区的起始地址映射到自己的虚拟地址空间中,两个进程就能看到同一份资源,从而能实现通信!!!

那为什么要个操作系统申请内存,而不给进程自己管理呢?
因为操作系统要对各种共享内存进行先描述,再组织的工作。

所以,共享内存一定有对应的描述该共享内存的对象,保存共享内存及其周边的各种属性和信息。
在这里插入图片描述

操作系统对这些对象进行管理的过程,本质转化成对链表的增删查改。


详谈共享内存的实现过程

二、共享内存的接口函数

1.shmget

shmget - allocates a System V shared memory segment
int shmget(key_t key, size_t size, int shmflg);

该接口就是向内存申请一块共享内存。


参数2:size

该参数就是申请的共享内存块的大小。

注意:一般申请的共享内存是4096字节(4KB)的整数倍。
如果申请的是4097字节,操作系统会给一块4096*2字节大小的共享内存,但是能够使用的只有4097字节,剩下的空间给了也不能用。

参数3:shmflg

这个参数类似于open函数的第三个参数,打开的方式:O_CREAT|O_WRONLY,shmflg参数的底层也是使用位图实现的。

在这里插入图片描述
重点是这两个宏定义

  • 1.IPC_CREAT单独使用时,如果不存在,就创建并返回,如果存在,就获取并返回。
  • 2.IPC_CREAT|IPC_EXCL一起使用时,如果不存在,就创建并返回,如果存在,则出错返回。
  • 3.IPC_EXCL不单独使用

第二点让人奇怪,解释如下:
IPC_CREAT|IPC_EXCL能保证如果能申请到,那么申请到的共享内存是最新的!


参数3:key

key是一个唯一标识符,也就是说每个共享内存都有唯一的key。

ftok  - convert a pathname and a project identifier to a System V IPC key
SYNOPSIS
#include <sys/types.h>
#include <sys/ipc.h>
key_t ftok(const char *pathname, int proj_id);

用户通过传递一个路径名和一个id,返回一个共享内存的唯一标识符。
所以ftok函数的本质就是一个算法。
pathname和proj_id是用户自己控制的。

为什么不让操作系统随机生成呢?
因为操作系统随机生成的key并不能传递给另一个进程,从而让不同的进程看到同一份资源这个目的。

所以必须让用户传参下来。

key_t key = ftok(pathname.c_str(),proj_id); // 成功返回key,失败返回-1

返回值:
在这里插入图片描述
如果共享内存申请成功,返回的是shmid,其实这个返回值就像是文件fd,创建一个文件,返回该文件在文件数组fd_array中的下标。申请失败返回-1.

  key_t key = ftok(PATH_NAME,proj_id);flag = IPC_CREATE|IPC_EXCL|0666;int shmid = shmget(key,size,flag); // 申请成功返回id,失败返回-1

所以可见,共享内存的确是由操作系统管理起来的。

所以,共享内存的生命周期是随操作系统的,进程退出共享内存并不会释放。除非内核重启,否则共享内存是不会释放的。


对比shmid和key:
shmid是共享内存在数组中的下标,只在进程内,用于标识资源的唯一性。
而key是内核级标定共享内存唯一性的。


共享内存的权限问题:
共享内存的权限,可以直接在shmget函数的第二个参数中传递。


如何保证让不同的进程看到同一份内存呢?


2. shmat

在这里插入图片描述

该函数是将指定进程与共享内存进行挂接。
第一个参数就是共享内存的id,第二个参数暂时不用管,设置为nullptr即可,第三个参数同样暂时不管,设置成0.

 //  2. 将服务端与共享内存挂接起来
char *shmaddr = (char *)shmat(shmid, nullptr, 0); 
// 返回挂接的虚拟地址的起始地址

shmdt

在这里插入图片描述
将挂接时获取的地址传过去,取消挂接即可,成功返回0,失败返回-1.

shmctl

在这里插入图片描述
就是将共享内存删除。
参数1传对应的共享内存,参数2传IPC_RMID,参数3先不管,穿nullptr;
参数2的命令如下,就是标记对应的共享内存为删除状态。
在这里插入图片描述

进程间使用共享内存通信

假设进程A申请共享内存。

对进程A来说:

  • 1.进程A先调用shmget函数,创建共享内存。
  • 2.进程A与对应的共享内存挂接起来。
  • 3.通信完成后取消挂接。
  • 4.再将共享内存释放。

对进程B来说:
1.进程B先调用shmget函数,获取共享内存。
2.进程B与对应的共享内存挂接起来。
3.通信完成后取消挂接。

三、共享内存的特性

1.共享内存没有同步互斥之类的保护机制

2.共享内存是所有进程间通信中,速度最快的!(拷贝少)
进程想向内存中写入数据,直接向对应的共享内存进行写入即可,只需要将用户层缓冲区拷贝到内存中即可。只需要一次拷贝。

3.共享内存内部的数据,由用户自己维护!!

关于代码

代码地址请移步:gitee——共享内存

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

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

相关文章

C语言精华题目锦集1

第一题 test.c文件中包括如下语句&#xff0c;文件中定义的四个变量中&#xff0c;是指针类型的是&#xff08;&#xff09;【多选】 #define INT_PTR int* typedef int* intptr; INT_PRT a,b; int_ptr c,d;A:a  B:b  C:c  D:d #define是宏定义&#xff0c;此时在程序中IN…

SQLite3 数据库学习(六):Qt 嵌入式 Web 服务器详解

参考引用 SQLite 权威指南&#xff08;第二版&#xff09;SQLite3 入门 1. Apache 搭建 cgi 环境 1.1 什么是 Apache Apache 是世界使用排名第一的 Web 服务器软件 它可以运行在几乎所有广泛使用的计算机平台上&#xff0c;由于其跨平台和安全性被广泛使用 1.2 具体搭建流程…

一、用户管理

一、后端数据库初始化 1.1 因为版本问题&#xff0c;始终报错&#xff0c;按照报错信息去查询解决方案&#xff0c;无法解决 灵机一动&#xff1a; 网址&#xff1a; Spring Boot 3.0 升级 实战踩坑记录 - 掘金 (juejin.cn) &#xff11;.&#xff12; 个人配置【运行成功…

c++的三目运算符

C三目运算符增强 C中的三目运算符表达式返回的可以是一个变量&#xff0c;但是C语言中返回的是一个常量。 C语言中&#xff1a; void test05() { int a 10; int b 20; printf("%d\n", a < b ? a : b); //在C语言中三目运算符返回的是表达式的值&am…

Javascript每天一道算法题(十三)——最大子数组和_中等

文章目录 动态规划题三个重要步骤&#xff08;了解思路&#xff09;1、问题2、示例3、解决方法&#xff08;1&#xff09;方法1——动态规划 总结 动态规划题三个重要步骤&#xff08;了解思路&#xff09; &#xff08;1&#xff09;定义数组元素的含义 用一个数组来保存历史数…

2020年06月 Scratch(三级)真题解析#中国电子学会#全国青少年软件编程等级考试

Scratch等级考试(1~4级)全部真题・点这里 一、单选题(共25题,每题2分,共50分) 第1题 执行以下脚本后舞台上的角色将 ? A:先克隆自身,克隆体出现后被删除。 B:先克隆自身,克隆体出现后删除本体。 C:克隆出自身后本体与克隆体同时被删除。 D:克隆出自身后本体与克…

docker常用命令, 镜像版本的导入、导出并加载,打包镜像的命令

文章目录 docker常用命令&#xff1a;打镜像包&#xff1a;镜像版本的导入、导出并加载 docker常用命令&#xff1a; 打镜像包&#xff1a; ​ docker build -t calc:20230630 /home/apps/calc/docker/ 删除某个镜像的版本&#xff0c;allen_mysql的5.7版本 docker rmi all…

Redis深入理解-内核请求处理流程、数据传输协议

Redis 内核级请求处理流程 Redis Server 其实就是 Linux 服务器中的一个进程 主要还是下图的流程 应用先和 server 端建立 TCP 连接建立连接之后&#xff0c;server 端就会有一个与该客户端通信的 socket&#xff0c;客户端的读写请求发送到服务端的 socket那么通过 IO 多路…

分组背包问题学习笔记 AcWing 9. 分组背包问题

原题 有 N&#xfffd; 组物品和一个容量是 V&#xfffd; 的背包。 每组物品有若干个&#xff0c;同一组内的物品最多只能选一个。 每件物品的体积是 vij&#xfffd;&#xfffd;&#xfffd;&#xff0c;价值是 wij&#xfffd;&#xfffd;&#xfffd;&#xff0c;其中 …

PC8233(CC/CV控制)高耐压输入5V/3.4A同步降压电路内建补偿带恒流恒压输出

概述 PC8233&#xff08;替代CX8853&#xff09;是一款同步降压调节器,输出电流高达3.4A,操作范围从8V到32V的宽电源电压。内部补偿要求最低数量现成的标准外部组件。PC8233在CC&#xff08;恒定输出电流&#xff09;模式或CV&#xff08;恒定输出电压&#xff09;模式&#x…

【前端】前端监控⊆埋点

文章目录 前端监控分为三个方面前端监控流程异常监控常见的错误捕获方法主要是 try / catch 、window.onerror 和window.addEventListener 等。Promise 错误Vue 错误React 错误 性能监控用户行为监控常见的埋点方案来源 前端监控分为三个方面 异常监控&#xff08;监控前端页面…

基于element-ui后台模板,日常唠嗑

后面会补充github地址 文章目录 目录 文章目录 案例说明 1.引入库 2.创建布局组件 3.创建布局组件 4.菜单效果展示 5.创建顶部组件 5.创建顶部面包屑组件 6.创建内容区域组件 7.效果总览 7.布丁&#xff08;实现一些小细节&#xff09; 前言一、pandas是什么&#xff1f;二、使…

CentOS7中升级OpenSSL详细教程

文章目录 一. 引言二. 升级前的准备1.备份现有配置2. 检查系统版本3. 安装依赖 三. OpenSSL安装四. 验证 一. 引言 OpenSSL: 是用于保护数据安全的重要工具。它能提供加密&#xff0c;解密等多项功能。 然而&#xff0c;随着技术的发展和新的安全漏洞的出现&#xff0c;使用最…

管理类联考——英语二——备考 100 句涵盖所有词汇

全中 在海里的这个地区&#xff0c;熊猫们喜欢就着苏打碗豆喝茶。而大洋州的民兵则喜欢经过半岛&#xff0c;带着编剧本的公式上餐厅去。附件的电影院里有额外的歌剧和香蕉&#xff0c;这一时代的斑马们被外面的天线所吸引。实验室里的蟹想用它的肋骨去戳四肢象灯炮的小羊。但…

千梦网创:创业,一场游戏一场梦

创业这件事就好比一场养成类游戏&#xff0c;而我们自己就是游戏主角。 这个游戏有一个特殊之处在于&#xff1a;SSS级装备有穿戴等级设定&#xff0c;就算你氪重金买到了一把神器&#xff0c;自身阅历不够也根本无法发挥它的强大威力而只能当个装饰。 这就要求我们真正沉浸在…

催单开发信怎么写?外贸人如何写催单邮件?

年末催单开发信编写技巧&#xff1f;最有效的催单话术有哪些&#xff1f; 催单开发信成为了企业间日常沟通的重要一环。这些信件不仅有助于促进业务发展&#xff0c;还可加强供应链的协调&#xff0c;确保货物及时送达。蜂邮EDM将介绍如何写一封出色的催单开发信&#xff0c;以…

ubuntu20.04安装多版本cuda,切换版本

1. 安装cuda toolkit: 下载网站 https://developer.nvidia.com/cuda-11.3.0-download-archive 选择版本&#xff0c;这里选择11.3 wget https://developer.download.nvidia.com/compute/cuda/11.3.0/local_installers/cuda_11.3.0_465.19.01_linux.run给cuda权限: chmod x…

Linux加强篇001-部署Linux系统

目录 一、前言 1.1准备工具 1.2安装配置VM虚拟机 1.3安装软件 1.4系统初始化进程 1.5重置root密码 二、巩固练习 1&#xff0e;为什么建议读者在下载系统文件后先进行校验而不是直接安装呢&#xff1f; 2&#xff0e;使用虚拟机安装Linux系统时&#xff0c;为什么要先…

科技与艺术如何交织出“理想之家”?三星电视给出家电行业最优解答

作者 | 曾响铃 文 | 响铃说 理想的家&#xff0c;是什么样子? 关于这个问题&#xff0c;社交媒体上有形形色色的答案。很多人的梦中情屋是原木风、奶油色&#xff0c;点缀着绿意盎然的植物&#xff1b;还有一些人的Dream house是用全屋智能将科技感拉满&#xff0c;再配上打…

OpenStack云计算平台-计算服务

目录 一、计算服务概览 二、安装并配置控制节点 1、先决条件 2、安全并配置组件 3、完成安装 三、安装和配置计算节点 1、安全并配置组件 2、完成安装 四、验证操作 一、计算服务概览 使用OpenStack计算服务来托管和管理云计算系统。OpenStack计算服务是基础设施即服务…