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

【Linux】进程间通信——共享内存

在这里插入图片描述

文章目录

  • 共享内存(Shared Memory)
    • 什么是共享内存
    • 2. 共享内存的特点
    • 3.共享内存的主要函数
      • 3.1.shmget()
      • 3.2.shmat
      • 3.3.shmdt
      • 3.4.shmctl
  • 共享内存实现进程间通信
    • ShareMemory.hpp
    • Server.cc
    • Client.cc
  • 总结

共享内存(Shared Memory)

什么是共享内存

共享内存(Shared Memory)是一种 进程间通信(IPC) 机制,允许多个进程共享同一块物理内存,从而提高数据交换效率。相比其他 IPC 方式(如管道、消息队列等),共享内存具有 速度快、低开销 的优势,因为数据直接存储在内存中,而无需通过内核进行数据拷贝。

2. 共享内存的特点

  • 高效:数据直接在内存中共享,避免了进程间数据拷贝的开销。
  • 进程可见:多个进程可以同时访问同一块共享内存,实现高速数据传输。
  • 需要同步机制:由于多个进程可以并发访问共享内存,通常需要使用 信号量(Semaphore)互斥锁(Mutex) 来防止数据竞争。

3.共享内存的主要函数

函数作用
shmget()创建或获取一个共享内存段
shmat()将共享内存附加到进程地址空间
shmdt()解除共享内存与进程的关联
shmctl()控制共享内存(删除、修改权限等)

3.1.shmget()

在这里插入图片描述
shmget表示获取共享内存,第一个参数key表示共享内存的键值,用于标识唯一的共享内存段。


这个参数由用户个人设置,但是通常用ftok函数来获取key。
在这里插入图片描述
ftok函数通过一定的算法来获取相对不会重复的key值,第一个参数是路径,第二个参数随机填一个数,通过算法获取相对唯一的key值。


shmget的第二个参数表示共享内存的大小,第三个参数表示标志位,如何创建共享内存和设置共享内存的权限。
在这里插入图片描述
第三个参数有特定的宏可以选择,红框框起来的是常用的两个。
IPC_CREAT:单独使用表示获取共享内存,如果存在则报错
IPC_CREAT | IPC_EXCL:表示创建共享内存
IPC_EXCL:单独使用没有意义

3.2.shmat

当我们获取到共享内存的时候,我们需要将共享内存挂接到虚拟内存地址当中,这时就需要用到这个接口。
在这里插入图片描述

用下面简图表示挂接:
在这里插入图片描述
shmget的第一个参数shmid表示shmget的返回值,会返回一个shmid,第二个参数表示我们可以指定一个虚拟地址,挂接到指定的虚拟地址上,但是一般情况下我们都会默认使用分配的虚拟地址,所以第二个参数一般情况下都会填nullptr,第三个参数表示标志位,用于控制映射方式(常用 0 或 SHM_RDONLY)。

3.3.shmdt

去关联,和上一个关联恰好相反,一个是关联一个是去关联。
在这里插入图片描述

3.4.shmctl

这个函数是用于控制共享内存的,在命令行我们一般用ipcrm -m shmid这个命令来删除共享内存,但是在代码层面,我们一般用shmctl这个函数来控制共享内存,可以进行删除修改权限等操作。
在这里插入图片描述

第二个参数表示标志位进行什么操作,下面是可以进行的操作,红框框起来的,表示删除共享内存,我们可以用这个宏来实现删除共享内存。
在这里插入图片描述
第三个参数是获取共享内存的信息,放在一个结构体当中,如果我们不需要获取直接传nullptr
在这里插入图片描述
接口讲完了,接下来用共享内存实现进程间通信

共享内存实现进程间通信

ShareMemory.hpp

#pragma once
#include <iostream>
#include <string>
#include <sys/types.h>
#include <cstdlib>
#include <ctime>
#include <unistd.h>
#include <sys/ipc.h>
#include <sys/shm.h>
using namespace std;
const string gpath = "/home/llllyrics/112_class";
int gprojid = 0X6666;
//操作系统,申请空间,是按照块为单位的:4kb,1kb,2kb,4mb.......
int gshmsize = 4096;
mode_t gmode = 0600;//权限

const int CREATE = IPC_CREAT | IPC_EXCL | gmode;
const int GET = IPC_CREAT;


class ShareMemory
{
private:
void CreateShmHelper(int flag)
    {
        //创建key
        _key = ftok(gpath.c_str(),gprojid);
        if(_key < 0)//创建失败
        {
            cerr<<"ftok error"<<endl;
            return;
        }
        //让server创建共享内存&&获取
        //注意:共享内存也有权限!在应用层和文件关联度不大,但是在底层和文件关联度大
        _shmid = shmget(_key,gshmsize,flag);//创建
        if(_shmid < 0)//创建失败
        {
            cerr<<"shmget error"<<endl;
            return;
        }
    }
public:
    ShareMemory():_shmid(-1),_key(0),_addr(nullptr){}
    ~ShareMemory(){}

    void CreateShm()
    {
        if(_shmid == -1)
            CreateShmHelper(CREATE);
    }
    void GetShm()
    {
        CreateShmHelper(GET);
    }
    void AttachShm()
    {
        //将共享内存挂接到自己的地址空间当中
        _addr = shmat(_shmid,nullptr,0);//将共享内存挂接到自己的虚拟地址上。
        if((long long)_addr == -1)return;//挂接失败返回nullptr
    }
    void DetachShm()
    {
        if(_addr != nullptr)
        shmdt(_addr);
        cout<<"detach done:"<<endl;
    }
    void DeleteShm()
    {
        int n = shmctl(_shmid,IPC_RMID,nullptr);
        if(n < 0) 
        {
            cout<<"delete failed"<<endl;
            return;
        }
        cout<<"delete shm done"<<endl;
    }
    void* GetAddr()
    {
        return _addr;
    }
    void ShmMeta()
    {

    }
private:
    int _shmid;
    key_t _key;
    void *_addr;
};
ShareMemory shm;

Server.cc

#include "ShareMemory.hpp"
int main()
{
    shm.CreateShm();
    shm.AttachShm();
    //接收----IPC
    char* strinfo = (char*)shm.GetAddr();//获取服务器的虚拟地址

    while(true)
    {
        sleep(1);
        //打印共享地址中的内容
        printf("%s\n",strinfo);
        //
    }
    shm.DetachShm();
    shm.DeleteShm();
    return 0;
}

Client.cc

#include "ShareMemory.hpp"
int main()
{
    shm.GetShm();
    shm.AttachShm();
    //写入----IPC
    char* strinfo = (char*)shm.GetAddr();//获取客户端的虚拟地址
    char ch = 'A';
    while(ch <= 'Z')
    {
        sleep(1);
        strinfo[ch-'A'] = ch;//这里操作共享内存的时候为什么没有用系统调用?
        ch++;
    }
    shm.DetachShm();
    return 0;
}

总结

共享内存作为一种高效的进程间通信机制,因其直接在内存中操作数据,避免了数据拷贝,提供了快速的数据传输方式。通过 shmgetshmatshmctl 等函数,Linux 系统为我们提供了灵活的共享内存操作接口。尽管共享内存具有显著的性能优势,但由于其没有内建的同步机制,使用时必须特别注意数据的一致性和进程间的同步问题。

在实际应用中,结合信号量、消息队列等同步机制,共享内存可以为多进程间提供高效且稳定的通信手段。然而,开发者需要注意资源的管理与清理,以免造成内存泄漏或数据冲突。

总之,共享内存是一种非常强大的进程间通信工具,但使用时需要小心谨慎,确保数据同步和资源管理得当,才能充分发挥其优势。

相关文章:

  • 一周一个Unity小游戏2D反弹球游戏 - 移动的弹板(触屏版)
  • Python--模块(上)
  • IoT 测试:智能互联时代的质量保障
  • DeepSeek-V3后训练揭秘:如何通过监督微调和强化学习提升模型性能
  • 为什么要进行软件测试?
  • 将宇宙不同温度下的能量表现形式 类比为量子计算机的波函数解码过程
  • 《Qt动画编程实战:轻松实现头像旋转效果》
  • 人工智能 pytorch篇
  • 构建动态URL查询字符串以导出报警统计数据
  • C++:指针函数与函数指针
  • Grafana11.5.2 | 安装配置步骤(已成功安装)
  • AI开发利器:Anaconda
  • Golang快速上手01/Golang基础
  • 深入miniqmt:掌握创建交易对象的关键步骤
  • 异常c/c++
  • 原码反码补码计算以及按位取反操作
  • Linux 环境“从零”部署 MongoDB 6.0:mongosh 安装与数据操作全攻略
  • Node.js安装与学习的简单记录
  • C++三大特性之多态
  • Mission Planner MP地面站添加Cesium三维地图
  • 高飞已任南航集团党组副书记
  • 一种声音·阿甘本|即将到来的中世纪;“新”与“旧”……
  • 中国恒大披露清盘进展:要求债权人提交债权证明表
  • 新闻1+1丨城市,如何对青年更友好?
  • 车载抬头显示爆发在即?业内:凭借市场和产业链优势,国内供应商实现反超
  • 违法违规收集使用个人信息,爱奇艺、轻颜等65款App被点名