波奇学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-基础知识3

打包和压缩 zip 安装zip软件包 yum -y install zip unzip 压缩打包命令: zip -q -r -d -u 压缩包文件名 目录和文件名列表 -q:不显示命令执行过程-r:递归处理,打包各级子目录和文件-u:把文件增加/替换到压缩包中-d:从压缩包中删除指定的文件 解压:unzip 压缩包名 打包文件 把压缩包从服务器下载到本地 把压缩包上传到服务器(zip

Linux 网络编程 --- 应用层

一、自定义协议和序列化反序列化 代码: 序列化反序列化实现网络版本计算器 二、HTTP协议 1、谈两个简单的预备知识 https://www.baidu.com/ --- 域名 --- 域名解析 --- IP地址 http的端口号为80端口,https的端口号为443 url为统一资源定位符。CSDNhttps://mp.csdn.net/mp_blog/creation/editor

【Python编程】Linux创建虚拟环境并配置与notebook相连接

1.创建 使用 venv 创建虚拟环境。例如,在当前目录下创建一个名为 myenv 的虚拟环境: python3 -m venv myenv 2.激活 激活虚拟环境使其成为当前终端会话的活动环境。运行: source myenv/bin/activate 3.与notebook连接 在虚拟环境中,使用 pip 安装 Jupyter 和 ipykernel: pip instal

Linux_kernel驱动开发11

一、改回nfs方式挂载根文件系统         在产品将要上线之前,需要制作不同类型格式的根文件系统         在产品研发阶段,我们还是需要使用nfs的方式挂载根文件系统         优点:可以直接在上位机中修改文件系统内容,延长EMMC的寿命         【1】重启上位机nfs服务         sudo service nfs-kernel-server resta

【Linux 从基础到进阶】Ansible自动化运维工具使用

Ansible自动化运维工具使用 Ansible 是一款开源的自动化运维工具,采用无代理架构(agentless),基于 SSH 连接进行管理,具有简单易用、灵活强大、可扩展性高等特点。它广泛用于服务器管理、应用部署、配置管理等任务。本文将介绍 Ansible 的安装、基本使用方法及一些实际运维场景中的应用,旨在帮助运维人员快速上手并熟练运用 Ansible。 1. Ansible的核心概念

Linux服务器Java启动脚本

Linux服务器Java启动脚本 1、初版2、优化版本3、常用脚本仓库 本文章介绍了如何在Linux服务器上执行Java并启动jar包, 通常我们会使用nohup直接启动,但是还是需要手动停止然后再次启动, 那如何更优雅的在服务器上启动jar包呢,让我们一起探讨一下吧。 1、初版 第一个版本是常用的做法,直接使用nohup后台启动jar包, 并将日志输出到当前文件夹n

[Linux]:进程(下)

✨✨ 欢迎大家来到贝蒂大讲堂✨✨ 🎈🎈养成好习惯,先赞后看哦~🎈🎈 所属专栏:Linux学习 贝蒂的主页:Betty’s blog 1. 进程终止 1.1 进程退出的场景 进程退出只有以下三种情况: 代码运行完毕,结果正确。代码运行完毕,结果不正确。代码异常终止(进程崩溃)。 1.2 进程退出码 在编程中,我们通常认为main函数是代码的入口,但实际上它只是用户级

【Linux】应用层http协议

一、HTTP协议 1.1 简要介绍一下HTTP        我们在网络的应用层中可以自己定义协议,但是,已经有大佬定义了一些现成的,非常好用的应用层协议,供我们直接使用,HTTP(超文本传输协议)就是其中之一。        在互联网世界中,HTTP(超文本传输协议)是一个至关重要的协议,他定义了客户端(如浏览器)与服务器之间如何进行通信,以交换或者传输超文本(比如HTML文档)。

如何编写Linux PCIe设备驱动器 之二

如何编写Linux PCIe设备驱动器 之二 功能(capability)集功能(capability)APIs通过pci_bus_read_config完成功能存取功能APIs参数pos常量值PCI功能结构 PCI功能IDMSI功能电源功率管理功能 功能(capability)集 功能(capability)APIs int pcie_capability_read_wo