Linux--进程间的通信-共享内存

这篇具有很好参考价值的文章主要介绍了Linux--进程间的通信-共享内存。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

前文:
Linux–进程间的通信-匿名管道
Linux–进程间的通信–进程池
Linux–进程间的通信-命名管道

共享内存

对于两个进程,通过在内存开辟一块空间(操作系统开辟的),进程的虚拟地址通过页表映射到对应的共享内存空间中,进而实现通信

特点和作用:

  1. 高效性: 共享内存是一种高效的进程间通信方式,因为它允许多个进程直接访问同一块内存,而无需进行复制或数据传输。
  2. 快速通信: 由于共享内存直接映射到进程的地址空间,因此读写速度快,适用于对通信速度有较高要求的场景。
  3. 灵活性: 共享内存提供了一种灵活的通信方式,允许多个进程在需要时访问共享数据,而无需通过中间介质进行通信。
  4. 数据共享: 多个进程可以通过共享内存实现数据共享,从而实现对数据的共同读写和处理。

模拟实现

Linux--进程间的通信-共享内存,Linux,linux,数据库,运维
Linux--进程间的通信-共享内存,Linux,linux,数据库,运维

代码

Comm.hpp:包含共享内存的创建,销毁,挂接进程等。

#pragma once

#include<stdio.h>
#include<iostream>
#include<string>
#include<cerrno>
#include<cstring>
#include<cstdlib>
#include<sys/ipc.h>
#include<sys/types.h>
#include<sys/shm.h>
using namespace std;


const char* pathname="/home/ubuntu/Learning/Pipe";
const int proj_id=0x66;

//在内核中,共享内存的基本单位是4kb,我们申请的大小相当于是n*4kb
const int DefaultSize=4096;

//将key值转换为16进制的;
string ToHEX(key_t k)
{
    char buffer[1024];
    snprintf(buffer,sizeof(buffer),"0x%x",k);
    return buffer;
}

//获取键值
key_t GetShmKeyorDie()
{
    key_t k=ftok(pathname,proj_id);
    if(k<0)
    {
        //当返回值为-1时,错误表示stat(2)系统调用错误
        cerr << "ftok error, errno : " << errno << ", error string: " << strerror(errno) << endl;
        exit(1);
    }
    return k;
}

//创建共享内存,只在该函数内调用
int CreateShmOrDie(key_t key,int size,int flag)
{
    int shmid = shmget(key,size,flag);
    if(shmid<0)
    {
        std::cerr << "shmget error, errno : " << errno << ", error string: " << strerror(errno) << std::endl;
        exit(2);
    }
    return shmid;

}

//调用时的创建共享内存
int CreateShm(key_t key,int size)
{
    //如果已经存在了,那么会报错;
    return CreateShmOrDie(key,size,IPC_CREAT|IPC_EXCL|0666);
}
//调用时的获取
int GetShm(key_t key,int size)
{
    return CreateShmOrDie(key,size,IPC_CREAT);
}

//删除共享内存
void DeleteShm(int shmid)
{
    int n=shmctl(shmid,IPC_RMID,nullptr);
    if(n<0)
    {
        cerr<<"shmctl error"<<endl;
    }
    else
    {
        cout<<"shmctl delete shm success, shmid: "<<shmid<<endl;
    }
}

//查看共享内存的状态
void ShmDebug(int shmid)
{
    struct shmid_ds shmds;
    int n=shmctl(shmid ,IPC_STAT,&shmds);
    if(n<0)
    {
        std::cerr << "shmctl error" << std::endl;
        return;
    }
    std::cout << "shmds.shm_segsz: " << shmds.shm_segsz << std::endl;
    std::cout << "shmds.shm_nattch:" << shmds.shm_nattch << std::endl;
    std::cout << "shmds.shm_ctime:" << shmds.shm_ctime << std::endl;
    std::cout << "shmds.shm_perm.__key:" << ToHEX(shmds.shm_perm.__key) << std::endl;
}

void* ShmAttach(int shmid)
{
    void* addr = shmat(shmid,nullptr,0);
    //第二个参数设置nullptr,表示让系统选择合适的地址进行连接
    if((long long int)addr==-1)
    {
        cerr<<"shmat error"<<endl;
        return nullptr;
    }
    return addr;
}

void ShmDetach(void* addr)
{
    int n=shmdt(addr);
    if(n<0)
    {
        cerr<<"shmdt error"<<endl;
    }
}

fifo.hpp:利用管道来实现对共享内存实现同步机制。

#include<iostream>
#include<string>
#include<cstring>
#include<cerrno>
#include<sys/types.h>
#include<sys/stat.h>
#include<unistd.h>
#include<fcntl.h>
#include<assert.h>
using namespace std;

#define Mode 0666
#define Path "./fifo"

class fifo
{
public:
    fifo(const string & path=Path)
    :_path(path)
    {
        umask(0);
        int n=mkfifo(_path.c_str(),Mode);
        if(n==0)
        {
            cout<< "mkfifo success" << endl;
        }
        else
        {
            cerr << "mkfifo failed, errno: " << errno << ", errstring: " << strerror(errno) << endl;
        }
    }
    ~fifo()
    {
        int n=unlink(_path.c_str());
        if (n == 0)
        {
            cout << "remove fifo file " << _path << " success" << endl;
        }
        else
        {
            cerr << "remove failed, errno: " << errno << ", errstring: " << strerror(errno) << endl;
        }
    }
private:
    string _path; //文件路径
};

class Sync
{
public:
    Sync()
    :_rfd(-1),
    _wfd(-1)
    {}
    void OpenReadOrDie()
    {
        _rfd=open(Path,O_RDONLY);
        if(_rfd<0)
            exit(1);
    }
    void OpenWriteDie()
    {
        _wfd=open(Path,O_WRONLY);
        if(_wfd<0)
            exit(1);
    }
    bool Wait()
    {
        bool ret=true;
        uint32_t c=0;
        ssize_t n=read(_rfd,&c,sizeof(uint32_t));
        if(n==sizeof(uint32_t))
        {
            cout<<"server wakeup ,begin read shm..."<<endl;
        }
        else if(n==0)
        {
            ret=false;
        }
        else
        {
            return false;
        }
        return ret;
    }
    void Wakeup()
    {
        uint32_t c=0;
        ssize_t n=write(_wfd,&c,sizeof(c));
        assert(n==sizeof(uint32_t));

        cout<<"wakeup server..."<<endl;
    }
    ~Sync() {}
private:
    int _wfd;
    int _rfd;
};

ShmServer.cc

#include "Comm.hpp"
#include "fifo.hpp"
#include<unistd.h>
int main()
{
    
    //1.获取key
    key_t key = GetShmKeyorDie();
    std::cout << "key: " << ToHEX(key) << std::endl;
    // sleep(2);
    
    //2.创建共享内存
    int shmid = CreateShm(key, DefaultSize);
    std::cout << "shmid: " << shmid << std::endl;
     sleep(2);
  

    //4.将共享内存与进程挂接
    char* addr=(char*)ShmAttach(shmid);
    cout<<"Attach shm success, addr: "<<ToHEX((uint64_t)addr)<<endl;
   
    //0.先引入管道
    fifo ff;
    Sync syn;
    syn.OpenReadOrDie();

   //进行通信
     while(1)
     {
        if(!syn.Wait())break;
        cout<<"shm content: "<<addr<<endl;
     }

    ShmDetach(addr);
    std::cout << "Detach shm success, addr: " << ToHEX((uint64_t)addr) << std::endl;
     
     
    //3.删除共享内存
    DeleteShm(shmid);
    return 0;
}

ShmClient.cc

#include"Comm.hpp"
#include "fifo.hpp"
#include<unistd.h>
int main()
{
    key_t key = GetShmKeyorDie();
    std::cout << "key: " << ToHEX(key) << std::endl;
    // sleep(2);

    int shmid = GetShm(key, DefaultSize);
    std::cout << "shmid: " << shmid << std::endl;
    
    
    char* addr=(char*)ShmAttach(shmid);
    cout<<"Attach shm success, addr: "<<ToHEX((uint64_t)addr)<<endl;
    
    
    //通信
    memset(addr,0,DefaultSize);
    Sync syn;
    syn.OpenWriteDie();
    for(char c ='A';c<='Z';c++)
    {
        addr[c-'A']=c;
        sleep(1);
        syn.Wakeup();
    }

    ShmDetach(addr);
    std::cout << "Detach shm success, addr: " << ToHEX((uint64_t)addr) << std::endl;
    
    return 0;
}

解释

获取键值和创建共享内存

Linux--进程间的通信-共享内存,Linux,linux,数据库,运维

如果ftok函数返回失败时,我们就需要不断的尝试,对路径名和id值进行修改,直至成功。一般来说,有几种可能:

  • 1:如果传入的路径名不存在
  • 2:传入的路径名没有读取权限,无法读取该文件的索引节点号
  • 3:文件的索引节点超过了8位,即超过了一个字节的范围
  • 4:系统中已经使用了所有的IPC键值

Linux--进程间的通信-共享内存,Linux,linux,数据库,运维
Linux--进程间的通信-共享内存,Linux,linux,数据库,运维
Linux--进程间的通信-共享内存,Linux,linux,数据库,运维
Linux--进程间的通信-共享内存,Linux,linux,数据库,运维
Linux--进程间的通信-共享内存,Linux,linux,数据库,运维

删除共享内存

Linux--进程间的通信-共享内存,Linux,linux,数据库,运维
Linux--进程间的通信-共享内存,Linux,linux,数据库,运维

查看共享内存的状态

Linux--进程间的通信-共享内存,Linux,linux,数据库,运维
Linux--进程间的通信-共享内存,Linux,linux,数据库,运维
Linux--进程间的通信-共享内存,Linux,linux,数据库,运维

挂接进程

Linux--进程间的通信-共享内存,Linux,linux,数据库,运维
Linux--进程间的通信-共享内存,Linux,linux,数据库,运维
Linux--进程间的通信-共享内存,Linux,linux,数据库,运维
Linux--进程间的通信-共享内存,Linux,linux,数据库,运维

进入通信

Linux--进程间的通信-共享内存,Linux,linux,数据库,运维

协同机制

Linux--进程间的通信-共享内存,Linux,linux,数据库,运维
Linux--进程间的通信-共享内存,Linux,linux,数据库,运维
Linux--进程间的通信-共享内存,Linux,linux,数据库,运维文章来源地址https://www.toymoban.com/news/detail-858535.html

到了这里,关于Linux--进程间的通信-共享内存的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • Linux 共享内存mmap,进程通信

    进程间通信是操作系统中重要的概念之一,使得不同的进程可以相互交换数据和进行协作。其中,共享内存是一种高效的进程间通信机制,而内存映射(mmap)是实现共享内存的一种常见方法。 存储映射 I/O 是 一个磁盘文件 与 存储空间中的一个缓冲区相映射 。于是, 当从缓

    2024年02月13日
    浏览(28)
  • 【Linux】进程间通信——管道/共享内存

    进程间通信( Inter-Process Communication,简称IPC )是指不同进程之间进行数据交换和共享信息的机制和技术。在操作系统中,每个进程都是独立运行的,有自己的地址空间和数据,因此进程之间需要一种机制来进行通信,以便彼此协调工作、共享数据或者进行同步操作。 进程间

    2024年02月16日
    浏览(30)
  • 【Linux】进程间通信之共享内存

    共享内存比管道快哦~ 文章目录 前言 一、共享内存的实现原理 二、实现共享内存的代码 总结 共享内存区是最快的IPC形式。一旦这样的内存映射到共享它的进程的地址空间,这些进程间数据传递不再涉及到内核,换句话说是进程不再通过执行进入内核的系统调用来传递彼此的

    2024年02月03日
    浏览(35)
  • 进程间通信--共享内存详解【Linux】

    本文详细讲解了共享内存的原理和使用,并且通过实例代码角度来深度理解共享内存,下面就让我们开始吧。 数据传输:一个进程需要将它的数据发送给另一个进程 资源共享:多个进程之间共享同样的资源。 通知事件:一个进程需要向另一个或一组进程发送消息,通知它(

    2024年02月02日
    浏览(28)
  • (26)Linux 进程通信之共享内存(共享储存空间)

    共享内存是System V版本的最后一个进程间通信方式。 共享内存,顾名思义就是允许两个不相关的进程访问同一个逻辑内存,共享内存是两个正在运行的进程之间共享和传递数据的一种非常有效的方式。不同进程之间共享的内存通常为同一段物理内存。进程可以将同一段物理内

    2024年01月16日
    浏览(31)
  • 【hello Linux】进程间通信——共享内存

    目录 前言: 1. System V共享内存 1. 共享内存的理解 2. 共享内存的使用步骤 3. 共享内存的使用         1. 共享内存的创建         查看共享内存         2. 共享内存的释放         3. 共享内存的挂接         4. 共享内存的去挂接 4. 共享内存的使用示例 1. 两进

    2024年02月01日
    浏览(66)
  • 【Linux】进程间通信 -- system V共享内存

    共享内存区是最快的IPC形式。一旦这样的内存映射到共享它的进程的地址空间,这些进程间数据传递不再涉及到内核,换句话说是进程不再通过执行进入内核的系统调用来传递彼此的数据 理解: 进程间通信,是专门设计的,用来IPC 共享内存是一种通信方式,所有想通信的进程

    2024年02月16日
    浏览(33)
  • 【Linux】进程间通信——system V共享内存

    目录  写在前面的话 System V共享内存原理 System V共享内存的建立 代码实现System V共享内存 创建共享内存shmget() ftok() 删除共享内存shmctl() 挂接共享内存shmat() 取消挂接共享内存shmdt() 整体通信流程的实现          上一章我们讲了进程间通信的第一种方式 --- 管道,这一章我

    2024年02月14日
    浏览(29)
  • Linux学习记录——이십 进程间通信(2)共享内存

    system是一套标准,独立于文件系统之外,是系统专门为通信而设计出来的内核模块,称之为system V的IPC通信机制。 共享内存的主要做法也是让两个毫不相关的进程看到同一份资源。 进程的地址空间内,栈和堆区之间有个共享区,堆是向上增长,栈是向下增长,重合的那个地方

    2023年04月24日
    浏览(28)
  • 【Linux】进程间通信之共享内存/消息队列/信号量

    共享内存是通过让不同的进程看到同一个内存块的方式。 我们知道,每一个进程都会有对应的PCB-task_struct ,独立的进程地址空间,然后通过页表将地址映射到物理内存中。此时我们就可以让OS在内存中申请一块空间,然后将创建好的内存空间映射到进程的地址空间中,两个需

    2024年02月05日
    浏览(38)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包