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

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.lryc.cn/news/287752.html

相关文章:

  • 深入分析 Linux 网络丢包问题
  • web安全学习笔记【08】——算法1
  • 2024最新版Python 3.12.1安装使用指南
  • Oracle 经典练习题 50 题
  • PyTorch的衍生资源
  • 开源项目Git Commit规范与ChangeLog
  • 【原理图PCB专题】OrCAD Capture CIS关闭开始界面
  • 【Linux】Ubuntu的gnome切换KDE Plasma
  • Docker(九)Docker Buildx
  • Flink问题解决及性能调优-【Flink不同并行度引起sink2es报错问题】
  • 瑞_数据结构与算法_二叉搜索树
  • Linux 命令行访问名字中包含空格的文件或文件夹
  • Dart/Flutter工具模块:the_utils
  • 矩阵号:日入100+,八大提示词(Prompt)使用技巧
  • 爬虫工作量由小到大的思维转变---<第三十九章 Scrapy-redis 常用的那个RetryMiddleware>
  • 【MongoDB】mongodb安装及启动踩坑点
  • 动态规划——采矿的小奇【集训笔记】
  • wpf控件Expander集合下的像素滚动
  • docker 基础手册
  • 记一次SPI机制导致的BUG定位【不支持:http://javax.xml.XMLConstants/property/accessExternalDTD】
  • Kali如何启动SSH服务并实现无公网ip环境远程连接
  • 谷粒商城配置虚拟机
  • Java中文乱码浅析及解决方案
  • 【前端基础--3】
  • Obsidian笔记软件结合cpolar实现安卓移动端远程本地群晖WebDAV数据同步
  • 51单片机电子密码锁Proteus仿真+程序+视频+报告
  • [BSidesCF 2020]Had a bad day
  • [笔记]事务简介-springboot
  • 初识计算机网络 | 计算机网络的发展 | 协议初识
  • 【sgTree】自定义组件:加载el-tree树节点整棵树数据,实现增删改操作。