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

【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 系统为我们提供了灵活的共享内存操作接口。尽管共享内存具有显著的性能优势,但由于其没有内建的同步机制,使用时必须特别注意数据的一致性和进程间的同步问题。

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

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


http://www.kler.cn/a/591267.html

相关文章:

  • 从0到1彻底掌握Trae:手把手带你实战开发AI Chatbot,提升开发效率的必备指南!
  • vue3中用v-for循环出三个元素,绑定元素的ref并拿到这三个元素的ref属性
  • JavaScript 日期区间计算:全面解析与实战应用
  • 第一个vue项目
  • 工作记录 2017-02-06
  • Mysql 安装指南(小白入门)
  • 改变一生的思维模型【14】奥卡姆剃刀理论
  • 【408计算机网络-自顶向下-应用层】-简单描述概念PPT-中国科学技术大学-郑老师-计算机网络课程的深度复习资料
  • 台达PLC转太网转换的教程案例(台达DVP系列)
  • 【工业现场总线】控制网络的主要特点是?OSI参考模型的分层是?
  • C++与C的基本不同
  • 2025年Go语言面试中常见的50道面试题,涵盖基础语法、并发编程、数据结构、错误处理
  • K8s的部署
  • 【Linux编程】IPC之消息队列从踩坑到实战:核心原理、实战案例与C++封装详解(含完整代码)
  • Leetcode 刷题笔记1 图论part01
  • Java 大视界 -- Java 大数据在智能家居设备联动与场景自动化中的应用(140)
  • 【VS小知识】VS如何保存UTF8
  • python-列表的操作以及切片
  • Groove 清除环境变量,以防应用程序因为环境变量设置了错误的 Qt 插件路径而启动失败
  • OpenHarmony子系统开发 - 电话服务