当前位置: 首页 > news >正文

【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——共享内存

相关文章:

  • Fiddler 无法抓包手机 https 报文的解决方案来啦!!
  • 如何应用ChatGPT撰写、修改论文及工作报告,提供写作能力及优化工作??
  • 微软离Altman越近,离OpenAI就越远!
  • OpenAI 曾收到 AI 重大突破警告;半独立的 OpenAI 比与微软合并更好丨 RTE 开发者日报 Vol.91
  • 网络安全(黑客)-自学手册
  • Mistral 7B 比Llama 2更好的开源大模型 (四)
  • sql添加索引
  • python之pyqt专栏2-项目文件解析
  • macos端文件夹快速访问工具 Default Folder X 最新for mac
  • 深度学习之生成唐诗案例(Pytorch版)
  • 华为云之在Linux系统下安装可视化界面
  • 被动接受需求
  • 【正点原子STM32连载】第五十九章 T9拼音输入法实验(Julia分形)实验 摘自【正点原子】APM32F407最小系统板使用指南
  • 竞赛 题目:基于深度学习的手势识别实现
  • 周总结2023-11-24
  • TMUX设置鼠标滚轮滑动来浏览之前的前面内容
  • Python BDD 框架比较之 pytest-bdd vs behave
  • Linux安装Mysql详细教程(两种安装方法)
  • 如何使用rclone将腾讯云COS桶中的数据同步到华为云OBS
  • Docker实践笔记7:构建MySQL 8镜像
  • 中华人民共和国和巴西联邦共和国关于强化携手构建更公正世界和更可持续星球的中巴命运共同体,共同维护多边主义的联合声明
  • 新剧|《藏海传》定档,《折腰》《人生若如初见》今日开播
  • 王毅人民日报撰文:共商发展振兴,共建中拉命运共同体
  • 消费维权周报|上周涉手机投诉较多,涉拍照模糊、屏幕漏液等
  • 著名学者黄修己去世,享年90岁
  • 警方通报“网约车司机偷拍女乘客”:已被行政拘留