Linux进程间通信:system V共享内存

news2025/1/13 10:17:39

目录

一、什么是共享内存

1.1创建共享内存

1.2释放共享内存

1.2.1shmctl

1.2.2shmat

1.2.3 shmdt

二、共享内存的实现及使用

2.1ShmClient

2.2Shm_Server

2.3Fifo.hpp

2.4Comm.hpp


一、什么是共享内存

标准系统V也叫system V的本地通信方式一般有三种:

1、共享内存

2、消息队列

3、信号量

而博主此片文章主要从共享内存方面去进行使用及讲解

1.1创建共享内存

共享内存的系统调用,第一个参数是共享内存在内核中唯一性的标识,第二个参数是所要创建共享内存的大小 ,第三个参数有两个选项:IPC_CRREAT和IPC_EXCL,有以下三种组合使用方法:

IPC_CRREAT:如果共享内存不存在,就创建它,如果共享内存已经存在,直接获取它

IPC_EXCL:不能单独使用,没有意义

IPC_CRREAT|IPC_EXCL:如果共享内存不存在就创建,如果存在就出错返回。

第三种方法可以保证如果创建的共享内存是成功的那么它一定是一个新的共享内存。

创建成功则会返回共享内存的标识符,失败返回-1,错误码被返回表示错误原因。

共享内存在内核中同时可以存在很多个。而OS管理多个共享内存依旧是六字真言:先描述再组织

所以对共享内存的管理就变成了对描述共享内存的结构体的管理。

而第一个参数key要具有唯一性,而这个key值不一定要我们自己去创建,可以去调用系统调用ftok这个算法生成一个key值。ftok第一个参数是用户传入一个const char*的字符串,第二个参数是一个int类型。所以我们可以通过ftok在两个进程中通过同样的pathname和id来让两个进行看到同一块共享内存。

文件操作,一个进程打开一个文件,进程退出的时候,这个被打开的文件就会被系统自动释放掉。文件的生命周期随进程但是注意:共享内存,如果进程结束,用户没有主动释放它,则其会一直存在。共享内存的生命周期随内核。

1.2释放共享内存

ipcs

ipcs可以查出三种资源:消息队列、共享内存、信号量。

perms则是权限

ipcs -m//查看系统中指定用户创建的共享内存

ipcrm -m shmid//删除shmid所对应的共享内存

以上都是指令级别的操作,而在代码中就需要调用系统接口来进行删除。

1.2.1shmctl

对共享内存进行设置、获取、删除三种功能。

shmid就是共享内存的id

cmd表示你想对其进行什么操作

buf则是含有共享内存熟悉的sturct结构体

获取用IPC_STAT可以获取struct shmid_ds中的各各种属性信息

设置用IPC_SET

删除用IPC_RMID

1.2.2shmat

at的意思就是attach关联的意思,因为共享内存是属于操作系统的,所以我们需要将共享内存挂接到当前进程地址空间的共享区,此时就需要用到shmat。

第一个参数表示要将哪一个共享内存贴到当前进程地址空间,第二个参数代表用户指明将shm挂接到哪里,所以第二个参数就是个地址(可以nullptr省略),第三个参数shmflag代表的是在进行挂接时的形式一般默认设置为0,保证可以读写。

如果挂接成功,会返回在进程地址空间中所挂接的虚拟地址。失败返回-1。

挂接完成后可以拿着返回的地址直接访问共享内存。 

1.2.3 shmdt

dt即detach去关联的意思,将共享内存从进程地址空间中删除。将共享内存的虚拟地址传进来,就可以进行进程地址空间和共享内存的去关联。

不管是指令级还是代码级的操作, 最后对共享内存进行控制,用的都是shmid,类似于操作文件时使用的fd。

二、共享内存的实现及使用

共享内存和管道不同,它不提供进程间协同的任何机制,这是共享内存的缺点,会引起数据不一致。但是共享内存是所有进程间通信中速度最快的。这是其优点 。

所以需要用户来进行共享内存协调机制的实现。

2.1ShmClient

#include "Comm.hpp"
#include "Fifo.hpp"

int main()
{
    //1、获取key
    key_t key=GetShmKeyOrDie();
    cout<<"key: "<<tosix(key)<<endl;
    
    //2、创建共享内存
    int shmid=GetShm(key,defaultsize);//创建一个全新的共享内存
    cout<<"shmid: "<<shmid<<endl;

    char* addr=(char*)ShmAttach(shmid);
    cout<<"Attach shm sucess,addr: "<<tosix((uint64_t)addr)<<endl;

    memset(addr,0,defaultsize);
    
    //以读方式打开管道
    Sync syn;
    syn.OpenReadOrDie();
    
    //进程间通信
    for(char c='A';c<='Z';c++)
    {
        addr[c-'A']=c;
        sleep(1);
        syn.Wakeup();
    }

    ShmDetach(addr);
    cout<<"Detach shm sucess,addr: "<<tosix((uint64_t)addr)<<endl;
    sleep(5);

    return 0;
}

2.2Shm_Server

#include "Comm.hpp"
#include "Fifo.hpp"

int main()
{
    //1、获取key
    key_t key=GetShmKeyOrDie();
    cout<<"key: "<<tosix(key)<<endl;

    //2、创建共享内存
    int shmid=CreateShm(key,defaultsize);//创建一个全新的共享内存
    cout<<"shmid: "<<shmid<<endl;

    //ShmDebug(shmid);

    //4、挂接共享内存
    char* addr=(char*)ShmAttach(shmid);
    cout<<"Attach shm sucess,addr: "<<tosix((uint64_t)addr)<<endl;

    //0.为了弥补进程间的协同机制,先引入管道
    Fifo fifo;
    Sync syn;
    syn.OpenReadOrDie();

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


    ShmDetach(addr);
    cout<<"Detach shm sucess,addr: "<<tosix((uint64_t)addr)<<endl;


    //3、删除共享内存
    sleep(100);
    DeleteShm(shmid);

    return 0;
}

2.3Fifo.hpp

#ifndef __COMM_HPP__
#define __COMM_HPP__

//头文件
#include <cstring>
#include <string>
#include <cerrno>
#include <iostream>
#include<cassert>

//调用fifo所需的头文件+opean用到的头文件
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>//open
//调用unlink对管道进行析构的头文件
#include <unistd.h>

//define
#define Mode 0666
#define Path "./fifo"
using namespace std;

class Fifo
{
public:
    Fifo(const string &path=Path):_path(path)//创建有名管道
    {
        umask(0);
        int n=mkfifo(_path.c_str(),Mode);
        if(n==0)
        {
            cout<<"mkfifo sucess"<<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<<"sucess"<<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 OpenWriteOrDie()
    {
        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(uint32_t));
        assert(n==sizeof(uint32_t)); 
        cout<<"wakeup server..."<<endl;
    }
    ~Sync()
    {

    }
private:
    int rfd;
    int wfd;
};

#endif

2.4Comm.hpp

#pragma once

#include <unistd.h>
#include <iostream>
#include <cerrno>
#include <string>
#include <cstring>
#include <cstdlib>
//shmget 创建共享内存所需头文件
#include <sys/ipc.h>
#include <sys/shm.h>
//ftok 所需头文件
#include <sys/types.h>

using namespace std;

//创建一个共享内存

const char* pathname="/home/gaz";
const int proj_id=0x66;
const int defaultsize=4096;//单位是字节
//在内核中,共享内存大小是以4KB为基本单位的,只能用自己申请的大小,一般建议申请大小为N*4KB
//将标识符转换成16进制
string tosix(key_t k)
{ 
    char buffer[1024];
    snprintf(buffer,sizeof(buffer),"0x%x",k);
    return buffer;
}
//根据pathname和proj_id转换出一个key值
key_t GetShmKeyOrDie()
{
    key_t k=ftok(pathname,proj_id);
    if(k<0)
    {
        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)
    {
        cerr<<"shmget error,errno: "<<errno<<",error string: "<<strerror(errno)<<endl;
        exit(2);
    }
    return shmid;
}
//对于服务端,创建一个全新的共享内存
int CreateShm(key_t key,int size)
{
    // IPC_CREAT: 不存在就创建,存在就获取
    // IPC_EXCL: 没有意义
    // IPC_CREAT | IPC_EXCL: 不存在就创建,存在就出错返回
    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 sucess,shmid: "<<shmid<<endl;
    }
}

void ShmDebug(int shmid)
{
    struct shmid_ds shmds;
    int n=shmctl(shmid,IPC_STAT,&shmds);//IPC_STAT获取shmid_ds中的各种属性信息
    if(n<0)
    {
        cerr<<"shmctl error"<<endl;
        return;
    }
    cout<<"shmds.shm_segsz: "<<shmds.shm_segsz<<endl;
    cout<<"shmds.shm_segsz: "<<shmds.shm_nattch<<endl;
    cout<<"shmds.shm_segsz: "<<shmds.shm_ctime<<endl;
}

void* ShmAttach(int shmid)//将共享内存和进程地址空间进行挂接
{
    void* addr=shmat(shmid,nullptr,0);
    if((long long int)addr==-1)//因为机器是64位的,int是4字节会引起精度损失
    {
        cerr<<"shmat error"<<endl;
        return nullptr;
    }
    return addr;
}

void ShmDetach(void* addr)//将共享内存和进程地址空间进行去关联
{
    int n=shmdt(addr);
    if(n<0)
    {
        cerr<<"shm error"<<endl;
    }
}

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

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

相关文章

论文阅读】 ICCV-2021-3D Local Convolutional Neural Networks for Gait Recognition

motivation :现有方法方法无法准确定位身体部位&#xff0c;不同的身体部位可以出现在同一个条纹(如手臂和躯干)&#xff0c;一个部分可以出现在不同帧(如手)的不同条纹上。其次&#xff0c;不同的身体部位具有不同的尺度&#xff0c;即使是不同帧中的同一部分也可以出现在不同…

2024041702-计算机操作系统 - 死锁

计算机操作系统 - 死锁 计算机操作系统 - 死锁 必要条件处理方法鸵鸟策略死锁检测与死锁恢复 1. 每种类型一个资源的死锁检测2. 每种类型多个资源的死锁检测3. 死锁恢复 死锁预防 1. 破坏互斥条件2. 破坏占有和等待条件3. 破坏不可抢占条件4. 破坏环路等待 死锁避免 1. 安全状态…

06-beanFactoryPostProcessor的执行

文章目录 invokeBeanFactoryPostProcessors(beanFactory)invokeBeanFactoryPostProcessors(beanFactory, getBeanFactoryPostProcessors())invokeBeanDefinitionRegistryPostProcessors(currentRegistryProcessors, registry);invokeBeanFactoryPostProcessors(regularPostProc…

docker desktop实战部署oracle篇

1、前言 oracle数据库官方已提供现成的镜像&#xff0c;可以直接拿来部署了。 由于项目中需要使用oracle数据库的分表功能&#xff0c;之前安装的是standard版本&#xff0c;无奈只能重新安装。网上查了一番&#xff0c;使用的方法都比较传统老旧&#xff1a;下载安装包手动安…

Ps中 饱和度 和 自然饱和度 的区别?

1.饱和度&#xff08;Saturation&#xff09;&#xff1a;在Photoshop中&#xff0c;饱和度是一个全局性调整&#xff0c;它影响图像中所有颜色的鲜艳程度。当你增加饱和度时&#xff0c;所有的颜色都会变得更浓烈、更鲜艳&#xff1b;相反&#xff0c;减小饱和度会使图像整体变…

解决 git克隆拉取代码报SSL certificate problem错误

问题&#xff1a;拉取代码时报错&#xff0c;SSL证书问题:证书链中的自签名证书问题 解决&#xff1a;只需要关闭证书验证&#xff0c;执行下面代码即可&#xff1a; git config --global http.sslVerify "false" 再次拉取代码就可以了

怎样选择IT外包公司?需要注意什么?

随着网络化、数字化、智能化快速发展&#xff0c;一部分企业成立自己的IT部门&#xff0c;负责各个科室的网络安全&#xff0c;大部分企业把网络安全、数据安全&#xff0c;外包给专业的IT外包公司&#xff0c;既提升了办公效率&#xff0c;企业又能把主要精力放在发展核心业务…

(二刷)代码随想录第1天|704. 二分查找 27. 移除元素

704. 二分查找 704. 二分查找 - 力扣&#xff08;LeetCode&#xff09; 代码随想录 (programmercarl.com) 手把手带你撕出正确的二分法 | 二分查找法 | 二分搜索法 | LeetCode&#xff1a;704. 二分查找_哔哩哔哩_bilibili 给定一个 n 个元素有序的&#xff08;升序&#xff09…

Unity如何使用adb工具安装APK

1、下载adb工具 SDK 平台工具版本说明 | Android Studio | Android Developers (google.cn) 2、配置环境变量 把platform-tools的路径添加进去就行 打开cmd&#xff0c;输入adb&#xff0c;即可查看版本信息 3、使用数据线连接设备&#xff0c;查看设备信息&#xff08;…

后教培时代的新东方,正在找寻更大的教育驱动力?

近段时间&#xff0c;K12教育主要上市公司的阶段性业绩皆已出炉。从具体数据来看&#xff0c;随着时间推移&#xff0c;教培机构的转型之路已愈走愈顺。 财报显示&#xff0c;2023年12月1日-2024年2月29日&#xff0c;好未来实现营收4.3亿美元&#xff0c;同比增长59.7%&#…

C++ | Leetcode C++题解之第60题排列序列

题目&#xff1a; 题解&#xff1a; class Solution { public:string getPermutation(int n, int k) {vector<int> factorial(n);factorial[0] 1;for (int i 1; i < n; i) {factorial[i] factorial[i - 1] * i;}--k;string ans;vector<int> valid(n 1, 1);…

小程序支付的款项流转与到账时间

商家做小程序&#xff0c;最关心的是客户通过小程序下单支付的钱&#xff0c;是怎么样的流转状态以及最终到哪里。因此&#xff0c;本文将详细解析款项最终流向何处以及多久能够到账。 一、小程序支付的款项流向 当用户在小程序内完成支付后&#xff0c;款项并不会直接到达商…

数据结构学习——线性表、顺序表

1.线性表 线性表 &#xff08; linear list &#xff09; 是n个具有相同特性的数据元素的有限序列。 线性表是一种在实际中广泛使 用的数据结构&#xff0c;常见的线性表&#xff1a;顺序表、链表、栈、队列、字符串… 线性表在逻辑上是线性结构&#xff0c;也就说是连续的一…

多线程【阻塞队列】(生产者消费者模型代码实现)

阻塞队列 解耦合削峰填谷生产者消费者模型&#xff1a; 解耦合 削峰填谷 生产者消费者模型&#xff1a; 正常来说&#xff0c;wait通过notify唤醒&#xff0c;其他线程调用了take,在take的最后一步进行notify. package thread; class MyBlockingQueue{private String [] data…

jmeter控制器讲解

1&#xff0c;随机顺序控制器和随机控制器的区别&#xff1a;随机顺序控制器下所有的接口都会执行&#xff0c;只是执行顺序是随机的&#xff0c;随机控制器下所有的接口中随机执行一个接口&#xff0c;其余接口不执行。

idea 项目 修改项目文件名 教程

文章目录 目录 文章目录 修改流程 小结 概要流程技术细节小结 概要 原项目名 修改流程 关掉当前项目的idea页面 修改之后的文件名 重新打开idea。选择项目打开项目页面 技术细节 出现下面这个问题&#xff0c;可以参考作者新的一编文章idea开发工具 项目使用Spring框架开发解…

【iOS】-- 内存五大分区

【iOS】-- 内存五大分区 内存五大分区1.栈区优点&#xff1a; 2.堆区优点&#xff1a; 3.全局区4.常量区5.代码区 验证static、extern、const关键字比较1.static关键字static关键字的作用&#xff1a;全局静态变量局部静态变量 2.extern关键字对内的全局变量对外的全局变量 3.c…

[报错解决]SpringBoot子项目打jar包启动报 XXX--1.0-SNAPSHOT.jar中没有主清单属性

目录 报错信息解决原因原因分析解决方案 报错信息 解决 原因 在使用SpringBoot架构搭建父子工程时&#xff0c;使用IDEA可以正常启动&#xff0c;对子项目打成jar包后使用jar方式启动时&#xff0c;会报错xx.jar中没有主清单属性。 原因分析 原因主要是在使用jar方式启动时…

2024年第六届世界软件工程研讨会(WSSE 2024)即将召开!

2024年第六届世界软件工程研讨会&#xff08;WSSE 2024&#xff09;将于2024年9月13-15日在日本京都举行。软件工程领域的发展离不开各位专家学者和业界精英的共同努力和贡献。WSSE 2024将就软件工程领域的最新研究成果、实践经验和发展趋势进行深入交流和探讨&#xff0c;汇聚…

VxTerm使用教程:连接SSH服务端设备,什么是SSH

一、什么是SSH&#xff1f; <摘自百度> 安全外壳协议 SSH&#xff0c;即安全外壳协议&#xff08;Secure Shell&#xff09;&#xff0c;是一种网络协议&#xff0c;用于在计算机网络上提供安全的远程登录和命令执行功能。 SSH通过加密通信通道来保护数据传输&#xff0c…