Linux入门——09 共享内存

news2024/10/2 6:31:39

1.共享内存原理

OS内的每个进程都会有自己的内核结构(task_struct)和虚拟地址空间,通过页表与物理内存进程映射。

如果让两个不同的进程共享内存,首先就是在内存中申请一块空间(共享内存),

然后将建立好的内存映射到进程的虚拟地址空间上(进程和共享内存挂接)。

两个内存都映射统一块物理地址空间,就是看到了统一块资源,就建立了通讯。

如果未来不想通讯了,就取消进程与内存的映射关系(去关联),释放内存。

理解:

  1. 进程间通讯,是专门设计的,用来IPC
  2. 共享内存是一种通讯方式,所有想通讯的进程,都可以用
  3. OS一定会存在很多共享内存

概念:通过让不同内存,看到同一内存块的方式:共享内存

共享内存是最快的IPC形式

2.实现共享内存

2.1shmget获得共享内存函数

int shmget(key_t key, size_t size, int shmflg);

参数:

key:保证看到的是同一份内存,内进行唯一性标识。(就像身份证号是什么不重要,它能保证你的唯一性)

key可以通过ftok获取。只要保证路径和id进行计算得到同一个数字key,就能保证看到同一个共享内存

size:共享内存大小,共享内存大小,一般为4kb(4096字节)的整数倍 (系统分配是按内存块分配的,内存块的基本单元为4kb)//如果是4097内核会给你向上取整,给你8kb。但是当用ipcs查看,确实是4097,!!!要知道给你的和能用的是两码事。

shmflg:IPC_CREAT IPC_EXCL(本质是标志位)

IPC_CREAT:如果创建的共享内存不存在,创建,如果存在,获取之。默认0也是如果创建的共享内存不存在,创建,如果存在,获取之。

IPC_EXCL:无法单独使用,IPC_EXCL | IPC_CREAT:如果不存在创建,如果存在,报错返回。 -------》如果创建成功,一定是一个新的共享内存

返回值:

成功,返回一个共享内存的标识符,错误返回-1

int getShmHelper(key_t k,int flag)

{

    int shmid = shmget(k,MAX_SIZE,flag);

    if(shmid < 0)

    {

        std::cerr

        exit(2);

    }

    return shmid;

}

//创建共享内存

int createShm(key_t k)

{

    return getShmHelper(k , IPC_CREAT | IPC_EXCL | 0600); //注意这里创建的权限,如果不加0600,后面挂接的时候就没有读和写的权限

}

如果不加权限0600,创建出来的共享内存的权限就为0,下图正perms就为0

 

2.2 ftok函数(转化,将一个路径和项目标识符转化为一个System V 的IPC的key)

参数:key_t ftok(const char *pathname, int proj_id);

pathname:路径

proj_id:项目id,自己随便写的标识符就行

主要功能就是将路径和id进行计算得到一个数字key

返回值:成功返回key,失败返回-1,

2.2.1理解key的意义

  • OS一定会存在很多个共享内存
  • 申请一个空间,TODO
  • key是什么不重要,能进行唯一性标识重要

malloc一个多大的空间,而free只用一个申请的起始地址 -------》我怎么知道要释放多少个,因为每次申请的时候,都会多给你一些空间,用来维护这个空间,所以只用起始地址就可以了‘;

共享内存同样也是,它也要管理。先描述再组织-------》共享内存=物理内存块+共享内存的相关属性

创建共享内存的时候,如何保证共享内存在系统中的唯一的?--------key!就相当与门牌号

key在那?在共享内存的相关属性里

在内核中struct shm

{

        key_t key;

}

在我们调用ftok创建共享内存块的时候,就会传进去一个key,把本质是将key传入创间好的共享内存的属性key中。

当另一个进程要通讯的时候会根据key,遍历所能通讯的共享内存块。

总结:key是要通过shmget,设置进共享内存属性中的!用来标识共享内存在内核中的唯一性

2.2.1shmid

shmid vs key有点像fd vs inode

inode是每个文件的唯一编号,fd描述符可以通过整数,快速让上层找到对应的文件

shmid就是用特定的key标识特定的共享内存,通过整数让上层快速访问共享内存

为什么怎么麻烦呢?用一个不就行了,这样不便于解耦,就像在公司 你要有员工号,你这个员工号没了,并不影响你在国家的身份证号 ,互不干扰。

3.代码实现共享内存

3.1comm.hpp

#ifndef _COMM_HPP_
#define _COMM_HPP_

#include <iostream>
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/shm.h>
#include <cstring>
#include <cerrno>
#include <cstdlib>
#include <unistd.h>


#define PATHNAME "/home/lin/Desktop/Linux_learn/sharem"
#define PROJ_ID 0x66
#define MAX_SIZE 4096  //字节,一共4kb

//获得shmget所需要的key
key_t getKey()
{
    key_t key = ftok(PATHNAME,PROJ_ID);
    if(key < 0)
    {
        // cin,cout,cerr--->stdin,stdout,stderr---> 0,1,2
        std::cerr<< errno <<":"<< strerror(errno) << std::endl;
        exit(1);
    }

    return key;
}

int getShmHelper(key_t k,int flag)
{
    int shmid = shmget(k,MAX_SIZE,flag);
    if(shmid < 0)
    {
        std::cerr << errno <<":"<< std::strerror(errno) << std::endl;
        exit(2);
    }
    return shmid;
}


//创建共享内存
int createShm(key_t k)
{
    return getShmHelper(k , IPC_CREAT | IPC_EXCL);
}



//获得共享内存
int getShm(key_t k)
{
     return getShmHelper(k , IPC_CREAT /*0*/);
}


//删除共享内存
int delShm(int shmid)
{
    if(shmctl(shmid,IPC_RMID,nullptr)==-1)
    {
        std::cerr <<errno <<":" << std::strerror(errno) <<std::endl;
        return -1;
    }
    else
    {
        return 0;
    }
}


#endif

3.3.2shm_server.cc

#include "comm.hpp"

int main()
{
    key_t key = getKey(); 
    printf("0x%x\n",key); 

    int shmid =  createShm(key);
    printf("%d\n",shmid); 

    sleep(5);
    delShm(shmid);

    return 0;
}

3.3.3shm_client.cc

#include "comm.hpp"

int main()
{
    key_t key = getKey(); 
    printf("0x%x\n",key); 

    int shmid =  getShm(key);
    printf("%d\n",shmid); 

    return 0;
}

当我们第一次执行./sshm_server的时候是成功的,当第二次执行会报错

可以看出来,进程结束的时候,申请的共享内存资源依然没有释放。-----》共享内存的声明周期是随操作系统的,不是随进程的。

这是system V的共性(消息队列,信号量数组)

4.查看共享内存&删除共享内存指令

ipcs命令 //查看共享内存

ipcs -m // -m代表共享内存

ipcs -q //-q代表消息队列

ipcs -s //-s代表信号数组

ipcs(system V的消息队列,共享内存,信号数组)

ipcrm -m shmid //删除共享内存

ipcrm -q shmid //删除消息队列

ipcrm -s shmid //删除信号数组

5.shmctl共享内存删除函数

int shmctl(int shmid, int cmd, struct shmid_ds *buf);

参数:

        shmid:共享内存id

        cmd:控制命令:

                IPC_STAT:从内核中将共享内存的属性获取下来

                IPC_SET:将设置好的共享内存属性设置到内核中

                IPC_RMID:立即移除共享内存

        struct shmid_ds *buf:获取共性内存的属性到结构体中打印下来,

                不同的结构体或区不同的属性,一般设为nullptr

                        IPC_INFO

                        SHM_INFO

                        SHM_STAT

                        SHM_LOCK

                        SHM_UNLOCK

返回值:

当执行删除的时候,成功返回0,失败返回-1;

6.共享内存的关联

6.1shmat共享内存关联函数

void *shmat(int shmid, const void *shmaddr, int shmflg);
参数:
    int shmid:表示跟那个共享内存关联起来
    const void *shmaddr:将这块共享内存映射到那块共享空间当中(百分百不使用,不知道映射到哪,一般为nullptr)
    int shmflg:跟读写权限有关,默认为0,就可以
    
    
返回值:void *shmat:返回的是映射好的地址空间,将在虚拟地址空间的起始地址给返回,等价于malloc
    成功返回起始地址
    错误返回-1

举例
//给共享内存产生相关联

void *attchShm(int shmid)
{
    void * mem =shmat(shmid,nullptr,0);
    if((int)mem == -1)
    {
        std::cerr <<errno <<":" << std::strerror(errno) <<std::endl;
        exit(3);
    }
    return mem;

}

6.2shmdt在使用完共享内存的时候去关联,不要粗暴的直接删除

去掉映射关系,删除页表与地址空间和物理内存的关联

int shmdt(const void *shmaddr);   //detach
参数:
    const void *shmaddr:将这块共享内存映射到那块共享空间当中的关系去掉 ,是shmat的返回值
          
返回值:
    成功返回0
    错误返回-1

7.生命周期随内核

7.1优点

  • 所有通讯方式中最快的!因为共享内存是双方在内存中共享数据,数据只要在内存就被双方看到,大大减少拷贝次数。

7.2缺点 

  • 不给我们进行同步和互斥的,没有对数据进行任何保护!管道有(有了再读,写满停止)

8.创建共享内存并使用

#include <sys/types.h>
#include <sys/shm.h>
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <string.h>

int main(int argc, char const *argv[])
{
    // 1.得到key
    key_t key;
    key = ftok("./keytest",100);
    if(key < 0 )
    {
        printf("ftok error\n");
        return -1;
    }
    printf("key = %d\n",key);
    // 2.创建共享内存
    int shmid;
    shmid = shmget(key,512,0666);
    if(shmid < 0 )
    {
        printf("shmget error\n");
        return -1;
    }
    printf("shmid = %d\n",shmid);
    // 3.映射共享内存
    char *buffer;
    buffer = shmat(shmid,NULL,0);
    if(buffer < 0)
    {
        perror(buffer);
        return -1;
    }
    // 4.读
    printf("读:%s\n",buffer);


    return 0;
}

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

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

相关文章

Unity XR Interaction Toolkit 踩坑记录

1&#xff1a;按下 grap/select 键 物品直接飞到手上 2 按下 grap/select 键 物品一点点的想自己移动

《机器学习》—— AUC评估指标

文章目录 一、什么是AUC&#xff1f;1、什么是ROC曲线&#xff1f;2、ROC曲线的绘制 二、如何计算AUC的值三、代码实现AUC值的计算四、AUC的优缺点 一、什么是AUC&#xff1f; 机器学习中的AUC&#xff08;Area Under the Curve&#xff09;是一个重要的评估指标&#xff0c;特…

走进虚拟机逃逸技术之VMware Escape漏洞CVE-2023-20872复现

走进虚拟机逃逸技术之VMware Escape漏洞CVE-2023-20872复现 技术分享 技术分享 起初&#xff0c;为了学习虚拟机逃逸相关技术&#xff0c;也为了搞懂硬件虚拟化。于是请教了某巨佬后告诉我一本书&#xff0c;看完之后为了验证我理解到的硬件虚拟化及虚拟化逃逸原理是否正确&am…

图书管理系统详细设计

需求概述 按照需求分析文档中的规格要求&#xff0c;使用条形码扫描器进书、借书、还书&#xff0c;使得信息传递准确、流畅。同时&#xff0c;系统最大限度地实现易安装&#xff0c;易维护性&#xff0c;易操作性&#xff0c;运行稳定&#xff0c;安全可靠。 软件结构 系统由…

如何让虚拟机识别到宿主机的USB设备

我的实验环境&#xff1a; Windows宿主机VirtualBox虚拟化软件一个Linux虚机一个8G的USB磁盘 首先要让虚拟机能看到宿主机的USB设备&#xff0c;这是在VirtualBox中设置的。 选中虚机&#xff0c;右键选择“设置”菜单&#xff0c;再单击“USB设备”&#xff1a; 选中“启用…

Python | Leetcode Python题解之第365题水壶问题

题目&#xff1a; 题解&#xff1a; class Solution:def canMeasureWater(self, x: int, y: int, z: int) -> bool:if x y < z:return Falseif x 0 or y 0:return z 0 or x y zreturn z % math.gcd(x, y) 0

Alembic:python中数据库迁移的瑞士军刀

Alembic 简介 Alembic 是由 SQLAlchemy 的创始人 Mike Bayer 设计的一个数据库迁移工具。它不仅支持自动迁移脚本生成&#xff0c;还允许开发者手动编辑迁移脚本来满足特定的需求。Alembic 通过提供一个环境来跟踪数据库模式的变更历史&#xff0c;确保数据库的版本与应用代码…

推荐一个完全自由的目录设计网站

引言 如果我们能通过网站出一本书&#xff0c;这将是一件很酷的事。 事实上&#xff0c;我们通过网站发布知识&#xff0c;最常见的是写博客。 这二者有什么区别呢&#xff1f; 书本的知识内容有很强的逻辑性、系统性。而博客是随心所欲的&#xff0c;一时灵感来了就写一篇…

关闭Chrome快捷键

chrome是没办法改变快捷键以及屏蔽快捷键的&#xff0c;需要安装插件&#xff1a;shortkey 保证插件是开启的 不用做其他设置所有快捷键已被关闭

OAPT:用于双JPEG伪影去除的偏移感知分区的Transformer

OAPT: Offset-Aware Partition Transformer for Double JPEG Artifacts Removal https://github.com/QMoQ/OAPT 2408.11480 (arxiv.org) 基于深度学习的方法在去除单个JPEG伪影任务中表现出了显著的性能。然而&#xff0c;现有方法在处理双重JPEG图像时往往会退化&#xff0c…

127-隧道搭建穿透上线FRPNPSNgrok

使用了几种工具将会一一介绍 ngrokru 项目地址&#xff1a;Sunny-Ngrok内网转发内网穿透 - 国内内网映射服务器 这个网站现在要实名认证&#xff08;还得花2元解锁&#xff09; 用这种在线的网站怎么说呢&#xff0c;真不如自己买个云服务器用下面的frp&#xff0c;毕竟流量…

Python3:多行文本内容转换为标准的cURL请求参数值

背景 在最近的工作中&#xff0c;经常需要处理一些接口请求的参数&#xff0c;参数来源形式很多&#xff0c;可能是Excel、知识库文档等&#xff0c;有些数据形式比较复杂&#xff0c;比如多行或者包含很多不同的字符&#xff0c;示例如下&#xff1a; **客服质检分析指引** …

多个程序监听不同网卡的相同端口、相同网卡不同IP的相同端口

1 概述 一个主机上的多个程序监听同一个端口&#xff0c;是否一定存在冲突&#xff1f;如果是多网卡、单网卡多IP的情景下&#xff0c;多个程序是可以独立监听的。 2 多个程序监听不同网卡的相同端口 3 多个程序监听同一个网卡不同IP的相同端口 4 小结 多个程序监听同一个网…

生成式人工智能会导致人工智能崩溃吗

况可能很快就会发生变化。 从定义上讲&#xff0c;LLM 需要大量数据&#xff0c;而且所使用的数据集越来越大。根据缩放定律[2]&#xff0c;要提高性能&#xff0c;必须同时增加参数数量和训练标记数量&#xff08;后者被认为是最重要的因素&#xff09;。 这些数据集包含人类产…

0x03 ShowDoc 文件上传漏洞(CNVD-2020-26585)复现

参考&#xff1a;ShowDoc文件上传漏洞&#xff08;CNVD-2020-26585&#xff09;_showdoc漏洞-CSDN博客 一、fofa 搜索使用该工具的网站 网络空间测绘&#xff0c;网络空间安全搜索引擎&#xff0c;网络空间搜索引擎&#xff0c;安全态势感知 - FOFA网络空间测绘系统 "S…

ZMQ请求应答模型

案例一 这个案例的出处是ZMQ的官网。请求段发送Hello&#xff0c;应答端回复World。 ZMQ Request(client) #include <string> #include <iostream> #include <zmq.hpp>using namespace std; using namespace zmq; // 使用 zmq 命名空间int main() {// ini…

知识竞赛答题设备及答题方式有哪些

根据我们多年的知识竞赛承办经验&#xff0c;我来谈谈在知识竞赛中常用的答题设备和答题方式。 一、常用答题设备 1.电脑 如果电脑资源充足&#xff0c;可以用笔记本电脑进行答题&#xff0c;笔记本电脑可以采取有线或无线方式进行连网&#xff0c;可以根据情况选择连网方案&…

PyTorch专栏介绍

专栏导读 深度学习作为人工智能领域的重要分支&#xff0c;其应用范围广泛&#xff0c;从图像识别到自然语言处理&#xff0c;再到强化学习等。PyTorch作为当前流行的深度学习框架之一&#xff0c;以其动态计算图和易用性受到了广大开发者的青睐。本专栏将带领读者从零开始&am…

kubeadm搭建生产环境高可用集群

前言 搞了好多天&#xff08;今天是20240819&#xff09;&#xff0c;中途遇到各种各样的问题&#xff0c;总算是可以用了 我这里用的vmware开了5台服务器做学习实践 K8S因为直接使用的 pkgs.k8s.io 仓库&#xff0c;所以直接拉取的最新release版&#xff08;v1.31&#xff09…

结构开发笔记(五):solidworks软件(四):绘制36x36方块摄像头基座

若该文为原创文章&#xff0c;转载请注明原文出处 本文章博客地址&#xff1a;https://hpzwl.blog.csdn.net/article/details/141422131 长沙红胖子Qt&#xff08;长沙创微智科&#xff09;博文大全&#xff1a;开发技术集合&#xff08;包含Qt实用技术、树莓派、三维、OpenCV…