波奇学Linux:共享内存

2024-03-01 07:04

本文主要是介绍波奇学Linux:共享内存,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

进程通信的前提:不同的进程看到同一份的资源

直接原理:同一块物理内存映射到不同进程的共享区

共享内存拆解:

1.申请内存,通过页表映射到进程地址空间

2.返回首地址,便于进程利用

3.释放共享内存,去关联

4.内存的申请必须要系统调用,确保空间不是私有的

5.利用先管理再组织,不难猜到内核有个struct结构体管理起来

申请共享内存系统调用接口

size_t 创建共享内存大小单位是字节,返回值为共享内存标识符,失败返回-1

shmflg参数选项:IPC_CREAT:如果存在返回,不存在就创建

IPC_CREAT|IPC_EXCL : 不存在就创建,存在就出错返回,保证申请的共享内存一个是新的

IPC_EXCL:不单独使用

参数key:作为某一块共享内存的标识符,拿到同一个key就可以看到同一块内存,key存在struct描述对象中

利用路径和项目id创建一个唯一key 用于申请共享内存

路径本身就具有唯一性!

实际运用时并不是直接拿到key,而是拿到pathname 和 proj_id

key vs shmid

key创建共享内存的参数,shmid 函数shmget的返回值

key操作系统内定唯一性 shmid只在进程内,表示资源唯一性

查看共享资源

当进程结束时,用户不主动关闭,共享内存依然存在,除非内核重启。

删除共享内存

ipcrm -m shmid

key是系统使用的,shmid是用户层使用。

挂接共享内存到地址空间:建立进程和共享空间的关系

返回值void* 共享空间在地址空间的虚拟地址

shmid:共享内存描述符

shamddr:挂接的地址 系统决定

shmflg:挂接的权限 保持权限不变,设为0

char* shamddr=(char*)shmat(shmid,nullptr,0);

 

 nattch等于1,表示有一个挂载当进程结束时会自动结束挂载,或者调用shmdt函数在进程就能结束挂载

shmdt(shamddr);

在进程中删掉共享内存

共享内存的属性由struct ipc_perm保存,猜测涉及到管理共享内存

cmd参数选项对共享内存的操作

shmctl(shmid,IPC_RMID,nullptr);

进程通过共享内存通信 

当两个进程同时挂接到共享内存时,两个进程实现通信。共享内存被挂载到进程的的地址空间,直接在进程中通过虚拟地址访问。

comm.hpp代码

#ifndef __COMM_HPP__
#define __COMM_HPP__#include<iostream>
#include<string>
#include<sys/ipc.h>
#include<cstring>
#include<sys/shm.h>
#include<sys/types.h>
#include"log.hpp"
using namespace std;
//共享内存大小一般建议4096的整数倍 1024字节=1kb
const int size=4096;
const string pathname="/home/boki";
const int proj_id=0x8888;
Log log;//获取key
//将GetKey和GetShoareMem都放在同一个头文件中保证获得的key相同
key_t GetKey()
{key_t k=ftok(pathname.c_str(),proj_id);if(k==-1){log(Fatal,"ftok error: %s",strerror(errno));exit(1);}log(Info,"ftok success,key is : %d",k);return k;
}int GetShareMemHelper(int flag)
{key_t k=GetKey();int shmid=shmget(k,size,flag);if(shmid<0){log(Fatal,"creat share memory error: %s",strerror(errno));exit(2);}log(Info,"create share memory success, shmid: %d",shmid);return shmid;
}
int CreateShm()
{return GetShareMemHelper(IPC_CREAT|IPC_EXCL|0666);
}
int GetShm()
{return GetShareMemHelper(IPC_CREAT);
}
#endif

process.a代码

#include "comm.hpp"using namespace std;
int main()
{// 创建共享内存int shmid=CreateShm();//共享内存挂接到进程char* shamddr=(char*)shmat(shmid,nullptr,0);while(true){cout<<"client say@"<<shamddr<<endl;sleep(1);}//删除挂接关系shmdt(shamddr);//删除共享内存shmctl(shmid,IPC_RMID,nullptr);return 0;
}

process.b代码

#include "comm.hpp"using namespace std;
int main()
{// 创建共享内存int shmid=GetShm();//共享内存挂接到进程char* shamddr=(char*)shmat(shmid,nullptr,0);// ipc codewhile(true){char buffer[1024];cout<<"Please Enter@ ";fgets(shamddr,4096,stdin);}//删除挂接关系shmdt(shamddr);return 0;
}

效果

 

此时可以把shamddr看成malloc返回的地址,且通信过程不再需要系统调用。

共享内存的机制特点

共享内存没有同步互斥保护机制:

          管道文件读端会阻塞等到写端打开,而共享内存两端各自独立

共享内存是所有的进程通信中,速度最快的:

        拷贝少,管道通信至少拷贝4次,用户-缓冲区-内核-缓冲区-屏幕

共享内存内部的数据,由自己保护

查看共享内存属性

内存文件同步互斥保护机制

通过管道的机制,每次写入信息到共享内存时,向管道传入符号,另一端管道文件接收到特殊符号,才允许从内存中读取。

这篇关于波奇学Linux:共享内存的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/761573

相关文章

Linux中SSH服务配置的全面指南

《Linux中SSH服务配置的全面指南》作为网络安全工程师,SSH(SecureShell)服务的安全配置是我们日常工作中不可忽视的重要环节,本文将从基础配置到高级安全加固,全面解析SSH服务的各项参... 目录概述基础配置详解端口与监听设置主机密钥配置认证机制强化禁用密码认证禁止root直接登录实现双因素

在Linux终端中统计非二进制文件行数的实现方法

《在Linux终端中统计非二进制文件行数的实现方法》在Linux系统中,有时需要统计非二进制文件(如CSV、TXT文件)的行数,而不希望手动打开文件进行查看,例如,在处理大型日志文件、数据文件时,了解... 目录在linux终端中统计非二进制文件的行数技术背景实现步骤1. 使用wc命令2. 使用grep命令

Linux如何快速检查服务器的硬件配置和性能指标

《Linux如何快速检查服务器的硬件配置和性能指标》在运维和开发工作中,我们经常需要快速检查Linux服务器的硬件配置和性能指标,本文将以CentOS为例,介绍如何通过命令行快速获取这些关键信息,... 目录引言一、查询CPU核心数编程(几C?)1. 使用 nproc(最简单)2. 使用 lscpu(详细信

linux重启命令有哪些? 7个实用的Linux系统重启命令汇总

《linux重启命令有哪些?7个实用的Linux系统重启命令汇总》Linux系统提供了多种重启命令,常用的包括shutdown-r、reboot、init6等,不同命令适用于不同场景,本文将详细... 在管理和维护 linux 服务器时,完成系统更新、故障排查或日常维护后,重启系统往往是必不可少的步骤。本文

基于Linux的ffmpeg python的关键帧抽取

《基于Linux的ffmpegpython的关键帧抽取》本文主要介绍了基于Linux的ffmpegpython的关键帧抽取,实现以按帧或时间间隔抽取关键帧,文中通过示例代码介绍的非常详细,对大家的学... 目录1.FFmpeg的环境配置1) 创建一个虚拟环境envjavascript2) ffmpeg-py

Linux脚本(shell)的使用方式

《Linux脚本(shell)的使用方式》:本文主要介绍Linux脚本(shell)的使用方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录概述语法详解数学运算表达式Shell变量变量分类环境变量Shell内部变量自定义变量:定义、赋值自定义变量:引用、修改、删

Linux链表操作方式

《Linux链表操作方式》:本文主要介绍Linux链表操作方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、链表基础概念与内核链表优势二、内核链表结构与宏解析三、内核链表的优点四、用户态链表示例五、双向循环链表在内核中的实现优势六、典型应用场景七、调试技巧与

详解Linux中常见环境变量的特点与设置

《详解Linux中常见环境变量的特点与设置》环境变量是操作系统和用户设置的一些动态键值对,为运行的程序提供配置信息,理解环境变量对于系统管理、软件开发都很重要,下面小编就为大家详细介绍一下吧... 目录前言一、环境变量的概念二、常见的环境变量三、环境变量特点及其相关指令3.1 环境变量的全局性3.2、环境变

Linux系统中的firewall-offline-cmd详解(收藏版)

《Linux系统中的firewall-offline-cmd详解(收藏版)》firewall-offline-cmd是firewalld的一个命令行工具,专门设计用于在没有运行firewalld服务的... 目录主要用途基本语法选项1. 状态管理2. 区域管理3. 服务管理4. 端口管理5. ICMP 阻断

Linux实现线程同步的多种方式汇总

《Linux实现线程同步的多种方式汇总》本文详细介绍了Linux下线程同步的多种方法,包括互斥锁、自旋锁、信号量以及它们的使用示例,通过这些同步机制,可以解决线程安全问题,防止资源竞争导致的错误,示例... 目录什么是线程同步?一、互斥锁(单人洗手间规则)适用场景:特点:二、条件变量(咖啡厅取餐系统)工作流