【Linux】—— 共享内存

news2024/11/16 1:28:05

本期我将要带大家学习的是有关进程间通信的另一种方式——共享内存。共享内存是一种用于进程间通信的高效机制,允许多个进程访问和操作同一块内存区域。

目录

(一)深刻理解共享内存

1.1 概念解释

1.2 共享内存原理

1.3 共享内存数据结构

1.4 共享内存函数

(二) 代码实现


(一)深刻理解共享内存

1.1 概念解释

  • 共享内存区是最快的IPC形式。一旦这样的内存映射到共享它的进程的地址空间,这些进程间数据传递不再涉及到内核,它允许不同的进程通过映射同一块物理内存区域来实现数据的直接读写,换句话说是进程不再通过执行进入内核的系统调用来传递彼此的数据

1.2 共享内存原理
 

  • 首先大家要明白要进入通信,前提是不是得有进程,而且对于共享内存它不像匿名管道那样必须得是父子进程;
  • 对于共享内存任意两个进程之间都可以创建,其中它的通信原理,我们在思考的时候一定要先想明白,只要此时存在一个进程它就有己的PCB、它就要有自己的地址空间、它就有自己的页表;
  • 同样的对于另一个进程,在被创建的时候也要有自己的PCB、也要有自己的地址空间、那么更要有自己的页表结构好。

【解释说明】 

  1. 不管是进程a还是B,这两进程当中任一个进程,它对应的都要将自己的数据,要能够映射到物理内存当中的特定区域;
  2. 最终它的地址空间当中,所有的代码和数据都会经过我们对应的页表映射到我们对应的物理内存当中,进而让我们找到该进程匹配的代码和数据;
  3. 对于进程B来讲它也有自己的地址空间,它也有自己的页表,最终可以映射到我们对应的内存当中的某一些特定的区域来代表它的代码或者数据;那么同样的进程a也有自己的代码和数据,包括它自己的,还有动态库等等,最后也会映射到我们物理内存当中的特定的区域

【解释说明】

  • 第一步:我们先有一段儿我们对应的物理内存的空间;
  • 第二步:对于进程A可以把一个库加载内存映射到他这个地址空间里,然后返回给用户,或者让对应的代码区的代码直接跳转到库里面去跑,而我们如果我创建了一块儿内存,然后想办法把对应的这块儿内存经过页表映射到我们对应的地址空间当中的某一个区域;然后我把这块儿地址空间对应的区域地址返回给用户。我们这里就相当于构建从共享内存到当前进程的地址空间内的映射,然后在这里将对应空间它的起始地址返回给用户;
  • 第三步:同样的进程B,它如果也做同样的工作也将对应的这部分地址空间映射到自己的那么共享区当中,将对应的代码和数据的那个虚拟起始地址,尤其是内存块儿的起始地址返回给用户,那么未来进程A和进程B只要使用对应的起始的虚拟地址就可以访问了。
  • 此时我们不就完成了让不同的进程A和进程B看到了同一份儿资源,这就叫做共享内存!!!

1.3 共享内存数据结构

struct shmid_ds {
    struct ipc_perm shm_perm; /* operation perms */
    int shm_segsz; /* size of segment (bytes) */
    __kernel_time_t shm_atime; /* last attach time */
    __kernel_time_t shm_dtime; /* last detach time */
    __kernel_time_t shm_ctime; /* last change time */
    __kernel_ipc_pid_t shm_cpid; /* pid of creator */
    __kernel_ipc_pid_t shm_lpid; /* pid of last operator */
    unsigned short shm_nattch; /* no. of current attaches */
    unsigned short shm_unused; /* compatibility */
    void *shm_unused2; /* ditto - used by DIPC */
    void *shm_unused3; /* unused */
};

1.4 共享内存函数

在Linux系统中,共享内存通常通过系统调用来实现。以下是一些与共享内存相关的主要函数:

ftok函数:

  • 功能:生成一个唯一的key,用于标识共享内存段。
  • 参数
    • keyfile:与路径名相关的文件名,可以是一个存在的文件。
    • id:一个非零整数。
  • 返回值:生成的key。

 man 手册查询结果:


shmget函数

功能:用来创建共享内存
原型:int shmget(key_t key, size_t size, int shmflg);
参数

  • key:这个共享内存段名字
  • size:共享内存大小
  • shmflg:由九个权限标志构成,它们的用法和创建文件时使用的mode模式标志是一样的

返回值:成功返回一个非负整数,即该共享内存段的标识码;失败返回-1

  man 手册查询结果:


shmat函数
功能:将共享内存段连接到进程地址空间
原型:void *shmat(int shmid, const void *shmaddr, int shmflg);
参数

  • shmid: 共享内存标识
  • shmaddr:指定连接的地址
  • shmflg:它的两个可能取值是SHM_RND和SHM_RDONLY

返回值:成功返回一个指针,指向共享内存第一个节;失败返回-1

  man 手册查询结果:

说明:

  •  shmaddr为NULL,核心自动选择一个地址
  • shmaddr不为NULL且shmflg无SHM_RND标记,则以shmaddr为连接地址。
  • shmaddr不为NULL且shmflg设置了SHM_RND标记,则连接的地址会自动向下调整为SHMLBA的整数倍。公式:shmaddr -(shmaddr % SHMLBA)
  • shmflg=SHM_RDONLY,表示连接操作用来只读共享内存

 shmdt函数

功能:将共享内存段与当前进程脱离
原型:int shmdt(const void *shmaddr);
参数

  • shmaddr: 由shmat所返回的指针

返回值:成功返回0;失败返回-1
注意:将共享内存段与当前进程脱离不等于删除共享内存段

  man 手册查询结果: 


shmctl函数
功能:用于控制共享内存
原型:int shmctl(int shmid, int cmd, struct shmid_ds *buf);
参数

  • shmid:由shmget返回的共享内存标识码
  • cmd:将要采取的动作(有三个可取值)
  • buf:指向一个保存着共享内存的模式状态和访问权限的数据结构

返回值:成功返回0;失败返回-1

   man 手册查询结果: 


(二) 代码实现

接下来,我们就用上述描述的接口简单的实现一下共享内存。 

【client.cc】

#include"common.hpp"
#include<unistd.h>
int main()
{
    key_t k = getkey();
    cout << "client key: " << toHex(k) << endl;

    int shmid = getShm(k, gsize);
    cout << "client shmid: " <<shmid<< endl;

    //3. 将自己和共享内存关联起来
    char* start = attachShm(shmid);

    sleep(15);
    
    // 4.  将自己和共享内存去关联
    detachShm(start);
    return 0;
}

【server.cc】

#include"common.hpp"
#include<unistd.h>
int main()
{
    //1. 创建key
    key_t k = getkey();
    cout << "server key: " << toHex(k) << endl;

    //2. 创建共享内存
    int shmid = createShm(k, gsize);
    cout << "server shmid: " << shmid << endl;

    sleep(3);

    //3. 将自己和共享内存关联起来
    char* start = attachShm(shmid);
    sleep(15);

    // 4. 将自己和共享内存去关联
    detachShm(start);

    //删除共享内存
    delShm(shmid);
    return 0;
}

【common.hpp】

#ifndef __COMMON_HPP__
#define __COMMON_HPP__

#include <iostream>
#include <cerrno>
#include <cstdio>
#include <cstring>
#include <cassert>
#include <string>
#include <sys/ipc.h>
#include <sys/shm.h>
#include <sys/types.h>
#include <sys/stat.h>
using namespace std;

#define PATHNAME "."
#define PROJID 0x6666
const int gsize = 4096; //暂时

key_t getkey(){
    key_t k =ftok(PATHNAME,PROJID);
    if(k == -1)
    {
        cerr << "error: " << errno << " : " << strerror(errno) << endl;
        exit(1);
    }
    return k;
}

string toHex(int x){
    char buffer[64];
    snprintf(buffer, sizeof buffer, "0x%x", x);
    return buffer;
}

static int createShmHelper(key_t k, int size, int flag){
    int shmid = shmget(k, gsize, flag);
    if(shmid == -1)
    {
        cerr << "error: " << errno << " : " << strerror(errno) << endl;
        exit(2);
    }
    return shmid;
}


int createShm(key_t k, int size){
    umask(0);
    return createShmHelper(k, size, IPC_CREAT | IPC_EXCL | 0666);
}

int getShm(key_t k, int size)
{
    return createShmHelper(k, size, IPC_CREAT);
}

void delShm(int shmid){
    int n = shmctl(shmid,IPC_RMID,nullptr);
    assert(n != -1);
    (void)n;
}

char* attachShm(int shmid)
{
    char *start = (char*)shmat(shmid, nullptr, 0);
    return start;
}

void detachShm(char *start)
{
    int n = shmdt(start);
    assert(n != -1);
    (void)n;
}

#endif

总的来说,共享内存是一种强大的进程间通信机制,适用于需要高效、频繁共享大量数据的场景。在使用时需要注意同步和互斥机制,以确保数据的一致性和安全性。

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

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

相关文章

【5G 接口协议】N2接口协议NGAP(NG Application Protocol)介绍

博主未授权任何人或组织机构转载博主任何原创文章&#xff0c;感谢各位对原创的支持&#xff01; 博主链接 本人就职于国际知名终端厂商&#xff0c;负责modem芯片研发。 在5G早期负责终端数据业务层、核心网相关的开发工作&#xff0c;目前牵头6G算力网络技术标准研究。 博客…

web开发学习笔记(13.mybatis基于注解配置)

1.使用mybatis基本步骤 2.引入依赖 <!-- mysql--><dependency><groupId>com.mysql</groupId><artifactId>mysql-connector-j</artifactId></dependency> <!-- mybatis--><dependency><groupId>org…

如何在Docker上运行Redis

环境: 1.windows系统下的Docker deckstop 1.Pull Redis镜像 2.运行Redis镜像 此时,Redis已经启动&#xff0c;我们登录IDEA查看下是否连接上了 显示连接成功&#xff0c;证明已经连接上Docker上的Redis了

制冷系统几种节流方式介绍

1.两次节流、中间气液分离 通过两个阶段的节流达到制冷剂的减压&#xff1a;通过第一毛细管&#xff0c;使液体制冷剂膨胀到中间压力后&#xff0c;由汽液分离器将气态制冷剂和液态制冷剂分离&#xff0c;液态制冷剂通过第二毛细管进一步膨胀&#xff08;减压&#xff09;&…

(十二)Head first design patterns代理模式(c++)

代理模式 代理模式&#xff1a;创建一个proxy对象&#xff0c;并为这个对象提供替身或者占位符以对这个对象进行控制。 典型例子&#xff1a;智能指针... 例子&#xff1a;比如说有一个talk接口&#xff0c;所有的people需要实现talk接口。但有些人有唱歌技能。不能在talk接…

一款自动化提权工具

免责声明 请勿利用文章内的相关技术从事非法测试&#xff0c;由于传播、利用此文所提供的信息而造成的任何直接或者间接的后果及损失&#xff0c;均由使用者本人负责&#xff0c;作者不为此承担任何责任。工具来自网络&#xff0c;安全性自测&#xff0c;如有侵权请联系删除。…

vue:element-ui表单动态验证规则

一、需求&#xff1a; 实现当是否发送消息选择是时&#xff0c;业务类型字段必填。 二、实现&#xff1a; 当你在一个表单中使用 el-form 和 el-form-item 来创建表单项时&#xff0c;el-form-item 的 :rules 属性可以用来设置该表单项的验证规则。我们希望根据用户在 "…

FPGA高端项目:Xilinx Zynq7020 系列FPGA纯verilog图像缩放工程解决方案 提供3套工程源码和技术支持

目录 1、前言版本更新说明给读者的一封信FPGA就业高端项目培训计划免责声明 2、相关方案推荐我这里已有的FPGA图像缩放方案本方案在Xilinx Kintex7 系列FPGA上的应用本方案在Xilinx Artix7 系列FPGA上的应用本方案在国产FPGA紫光同创系列上的应用本方案在国产FPGA高云系列上的应…

opencv#28 图像卷积

图像卷积 图像卷积是图像处理中最为基础的操作之一&#xff0c;其常用在图像的边缘检测&#xff0c;图像的去噪声以及图像压缩等领域。 图像卷积主要步骤: Step1:将卷积模板旋转180。 Step2:卷积模板移动到对应位置。 Step3:模板内求和&#xff0c;保存求和结果。 Step4:滑…

模拟器单窗口ip有问题?试试关闭IPV6来解决

目前应该不止雷电9有这个问题了&#xff0c;最早是看到无忧群里在说有这个问题&#xff0c;后面发现很多其他的ip软件也有同样的问题&#xff0c;很多人都遇到&#xff0c;所以做个图文教程在这里&#xff0c;没出问题的也可以设置一下&#xff0c;目前ipv6也还没普及&#xff…

HTML+CSS:3D轮播卡片

效果演示 实现了一个3D翻转的卡片动画&#xff0c;其中每个卡片都有不同的图片和不同的旋转角度。整个动画循环播放&#xff0c;无限次。整个页面的背景是一个占据整个屏幕的背景图片&#xff0c;并且页面内容被隐藏在背景图片之下。 Code <div class"container"…

【软件测试】学习笔记-JMeter 参数化策略

这篇文章将探讨JMeter重要的知识点&#xff1a;参数化。无论是从使用频率还是从参数化对性能测试结果的影响&#xff0c;它都是你做性能测试必须要掌握的。 参数化是什么 简单来说&#xff0c;参数化就是选取不同的参数作为请求内容输入。使用 JMeter 测试时&#xff0c;测试…

粘玉石用什么胶最好?

粘合玉石时&#xff0c;选择合适的胶水至关重要&#xff0c;因为不同的胶水有不同的特性&#xff0c;适用于不同类型的材料。对于玉石&#xff0c;一般建议使用以下类型的胶水&#xff1a; 1.透明环氧树脂胶&#xff1a; 透明的环氧树脂胶通常是粘合玉石的良好选择。它具有优秀…

上位机图像处理和嵌入式模块部署(windows opencv)

【 声明&#xff1a;版权所有&#xff0c;欢迎转载&#xff0c;请勿用于商业用途。 联系信箱&#xff1a;feixiaoxing 163.com】 opencv可以运行在多个平台上面&#xff0c;当然windows平台也不意外。目前来说&#xff0c;opencv使用已经非常方便了&#xff0c;如果不想自己编译…

PgSQL - 17新特性 - 块级别增量备份

PgSQL - 17新特性 - 块级别增量备份 PgSQL可通过pg_basebackup进行全量备份。在构建复制关系时&#xff0c;创建备机时需要通过pg_basebackup全量拉取一个备份&#xff0c;形成一个mirror。但很多场景下&#xff0c;我们往往不需要进行全量备份/恢复&#xff0c;数据量特别大的…

立创EDA学习:PCB布局

目前进度 ESP32最小系统板项目&#xff0c;已完成原理图绘制 点击“更新/转换原理图到PCB” 点击“应用修改” 对应器件的封装就可以对应到PCB中 布局传递 回到原理图&#xff0c;框选每一个模块&#xff0c;“设计-布局传递” 会跳转到PCB界面&#xff0c;可以自己选择放置位…

Linux - 安装字体库解决乱码问题

文章目录 问题描述步骤资源 问题描述 该安装方法&#xff0c;不区分中文和英文字体 Java在linux上转word文档为pdf&#xff0c; linux的字体缺失&#xff0c;导致了转出的pdf为乱码。 ● Linux将word转为pdf后出现乱码&#xff1f; ● 在linux上将word转为pdf 是乱码 ● 在lin…

http网络编程——在ue5中实现文件传输功能

http网络编程在ue5中实现 需求&#xff1a;在unreal中实现下载功能&#xff0c;输入相关url网址&#xff0c;本地文件夹存入相应文件。 一、代码示例 1.Build.cs需要新增Http模块&#xff0c;样例如下。 PublicDependencyModuleNames.AddRange(new string[] { "Core&q…

imgaug库图像增强指南(35):【iaa.Fog】——轻松创建自然雾气场景

引言 在深度学习和计算机视觉的世界里&#xff0c;数据是模型训练的基石&#xff0c;其质量与数量直接影响着模型的性能。然而&#xff0c;获取大量高质量的标注数据往往需要耗费大量的时间和资源。正因如此&#xff0c;数据增强技术应运而生&#xff0c;成为了解决这一问题的…

JavaEE进阶(6)SpringBoot 配置文件(作用、格式、properties配置文件说明、yml配置文件说明、验证码案例)

接上次博客&#xff1a;JavaEE进阶&#xff08;5&#xff09;Spring IoC&DI&#xff1a;入门、IoC介绍、IoC详解&#xff08;两种主要IoC容器实现、IoC和DI对对象的管理、Bean存储、方法注解 Bean)、DI详解&#xff1a;注入方式、总结-CSDN博客 目录 配置文件作用 Sprin…