进程间通信之共享内存

共享内存

  • 1.共享内存的概念
  • 2.共享内存函数
    • 2.1 shmget函数
    • 2.2 shmat函数
    • 2.3 shmdt函数
    • 2.4 shmctl函数
  • 3. 共享内存的使用

1.进程间通信的分类:
(1)管道:1、匿名管道pipe;2、命名管道mkfifo
(2)System V IPC:1、System V 消息队列;2、System V 共享内存;3、System V 信号量。
(3)POSIX IPC:1、消息队列;2、共享内存;3、信号量;4、互斥量;5、条件变量;6、读写锁。
前面已经了解了进程间管道通信,那么共享内存又是什么原理?

1.共享内存的概念

什么是共享内存?
共享内存通信是一种进程间通信的方式,它允许两个或更多进程访问同一块内存,就如同 malloc () 函数向不同进程返回了指向同一个物理内存区域的指针。共享内存是 Unix/Linux下的多进程之间的通信方法,这种方法通常用于一个程序的多进程间通信,实际上多个程序间也可以通过共享内存来传递信息。而且共享内存区是最快的IPC形式。一旦这样的内存映射到共享它的进程的地址空间,这些进程间数据传递不再涉及到内核,换句话说是进程不再通过执行进入内核的系统调用来传递彼此的数据。

共享内存和管道的区别:

管道通信和共享内存都是进程间通信的方式,但是它们的实现方式不同。管道通信需要在内核和用户空间进行四次的数据拷贝:由用户空间的buffer中将数据拷贝到内核中 -> 内核将数据拷贝到内存中 -> 内存到内核 -> 内核到用户空间的buffer。而共享内存是最快的一种 IPC,因为进程是直接对内存进行存取。多个进程可以同时操作,所以需要进行同步。

共享内存示意图:
在这里插入图片描述

共享内存数据结构:
用man shmctl指令可以查看共享内存的数据结构。

struct shmid_ds {struct ipc_perm shm_perm;    /* Ownership and permissions */size_t          shm_segsz;   /* Size of segment (bytes) */time_t          shm_atime;   /* Last attach time */time_t          shm_dtime;   /* Last detach time */time_t          shm_ctime;   /* Last change time */pid_t           shm_cpid;    /* PID of creator */pid_t           shm_lpid;    /* PID of last shmat(2)/shmdt(2) */shmatt_t        shm_nattch;  /* No. of current attaches */...
};

而其中ipc_perm是一个内核为每个IPC对象所维护的一个数据结构,如下:

struct ipc_perm {key_t          __key;    /* Key supplied to shmget(2) */uid_t          uid;      /* Effective UID of owner */gid_t          gid;      /* Effective GID of owner */uid_t          cuid;     /* Effective UID of creator */gid_t          cgid;     /* Effective GID of creator */unsigned short mode;     /* Permissions + SHM_DEST andSHM_LOCKED flags */unsigned short __seq;    /* Sequence number */
};

2.共享内存函数

2.1 shmget函数

shmget函数功能:用来创建共享内存。

NAMEshmget - allocates a System V shared memory segment//分配System V共享内存段
SYNOPSIS#include <sys/ipc.h>#include <sys/shm.h>int shmget(key_t key, size_t size, int shmflg);DESCRIPTIONshmget() returns the identifier of the System V shared memory segment associated with the value of the argument key.  A new shared memory seg?ment, with size equal to the value of size rounded up to a multiple of PAGE_SIZE, is created if key has the value  IPC_PRIVATE  or  key  isn'tIPC_PRIVATE, no shared memory segment corresponding to key exists, and IPC_CREAT is specified in shmflg.If shmflg specifies both IPC_CREAT and IPC_EXCL and a shared memory segment already exists for key, then shmget() fails with errno set to EEX?IST.  (This is analogous to the effect of the combination O_CREAT | O_EXCL for open(2).)The value shmflg is composed of:IPC_CREAT   to create a new segment.  If this flag is not used, then shmget() will find the segment associated with key and check  to  see  ifthe user has permission to access the segment.IPC_EXCL    used with IPC_CREAT to ensure failure if the segment already exists.RETURN VALUEOn success, a valid shared memory identifier is returned.  On errir, -1 is returned, and errno is set to indicate the error.//成功返回有效的共享内存标识符,失败返回-1,并且错误码被设置。

int shmget(key_t key, size_t size, int shmflg);
key_t key:这个值用ftok生成,ftok会经过算法生成出一个冲突概率低的值,这个值保证唯一;目的是申请的共享内存块是尽可能不同的;
size_t size:申请共享内存块的大小;
int shmflg:这个参数常用两个选项,分别是IPC_CREAT and IPC_EXCL;
单独使用IPC_CREAT: 创建一个共享内存,如果共享内存不存在,就创建之,如果已经存在,获取已经存在的共享内存并返回;
IPC_CREAT | IPC_EXCL: 创建一个共享内存,如果共享内存不存在,就创建之, 如果已经存在,则立马出错返回
IPC_EXCL不能单独使用,一般都要配合IPC_CREAT;
返回值:成功返回一个非负整数,即该共享内存段的标识码;失败返回-1。

ftok函数:

NAMEftok - convert a pathname and a project identifier to a System V IPC keySYNOPSIS#include <sys/types.h>#include <sys/ipc.h>key_t ftok(const char *pathname, int proj_id);RETURN VALUEOn success, the generated key_t value is returned.  On failure -1 is returned, with errno indicating the error as for the stat(2) system call.

ftok会将这个路径pathname和proj_id(可以随便写)经过算法生成出一个冲突概率低的值。

2.2 shmat函数

shmat函数功能:将共享内存段连接到进程地址空间。

void *shmat(int shmid, const void *shmaddr, int shmflg);
shmid: 共享内存标识;
shmaddr:指定连接的地址;
shmflg:它的两个可能取值是SHM_RND和SHM_RDONLY;
返回值:成功返回一个指针,指向共享内存;失败返回-1;
shmaddr为NULL,核心自动选择一个地址;
shmaddr不为NULL且shmflg无SHM_RND标记,则以shmaddr为连接地址;
shmaddr不为NULL且shmflg设置了SHM_RND标记,则连接的地址会自动向下调整为SHMLBA的整数倍;(所以一般直接设为nullptr就可)
shmflg=SHM_RDONLY,表示连接操作用来只读共享内存。

返回值为一个指针,并且指针指向共享内存,所以使用这个指针进行数据的写入或读出。

2.3 shmdt函数

shmdt函数功能:将共享内存段与当前进程脱离。

int shmdt(const void *shmaddr);
参数
shmaddr: 由shmat所返回的指针
返回值:成功返回0;失败返回-1
注意:将共享内存段与当前进程脱离不等于删除共享内存段

2.4 shmctl函数

shmctl函数功能:用于控制共享内存。

int shmctl(int shmid, int cmd, struct shmid_ds *buf);
参数
shmid:由shmget返回的共享内存标识码
cmd:将要采取的动作(有三个可取值),如下图所示;
buf:指向一个保存着共享内存的模式状态和访问权限的数据结构;
返回值:成功返回0;失败返回-1。

命令说明
IPC_STAT将shmid_ds结构中的数据设置为共享内存的当前关联值
IPC_SET在进程有足够权限的前提下,将共享内存的当前关联值设置为shmid_ds数据结构中给出的值
IPC_RMID删除共享内存段

3. 共享内存的使用

使用之前,先认识下面的IPC指令,共享内存,消息队列,信号量等指令基本相似,所以在使用共享内存,消息队列,信号量进行通信时,其都有一批函数,总的说是大同小异,但是原理是不同的。

查看命令删除命令
ipcs -m : 查看共享内存ipcrm -m shmid : 删除共享内存
ipcs -q : 查看共享内存ipcrm -q msqid : 删除消息队列
ipcs -s : 查看共享内存ipcrm -s semid : 删除信号量

comm.hpp代码如下:

#pragma once
#include <iostream>
#include <cerrno>
#include <cstring>
#include <unistd.h>
#include <cassert>
#include <sys/shm.h>
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/stat.h>
using namespace std;//  IPC_CREAT and IPC_EXCL
// 单独使用IPC_CREAT: 创建一个共享内存,如果共享内存不存在,就创建之,如果已经存在,获取已经存在的共享内存并返回
// IPC_EXCL不能单独使用,一般都要配合IPC_CREAT
// IPC_CREAT | IPC_EXCL: 创建一个共享内存,如果共享内存不存在,就创建之, 如果已经存在,则立马出错返回 -- 如果创建成功,对应的shm,一定是最新的!#define PATHNAME "."
#define PROJID 0x6666const int gsize = 4096; // 共享内存的大小key_t getKey()
{key_t k = ftok(PATHNAME, PROJID); //key_t ftok(const char *pathname, int proj_id);if(k == -1){cerr << "error: " << errno << " : " << strerror(errno) << endl;exit(1);}return k;
}

server.cc代码如下:

#include "comm.hpp"int main()
{//1.创建共享内存先要创建一个key_t k(ftok)key_t k = getKey();//2.创建一个共享内存(shmget)umask(0); //默认权限int shmid = shmget(k, gsize, IPC_CREAT | IPC_EXCL | 0666); //int shmget(key_t key, size_t size, int shmflg)// 因为server创建共享内存,所以第三个参数为IPC_CREAT | IPC_EXCL,这个共享内存一定是最新的if(shmid == -1){cerr << "error: " << errno << " : " << strerror(errno) << endl;exit(2);}//3.将共享内存段连接到进程地址空间(shmat)char* start = (char*)shmat(shmid, nullptr, 0);if(*start == -1){cerr << "error: " << errno << " : " << strerror(errno) << endl;exit(3);}//3.写入信息"i am process server"char buffer[64] = "i am process server";int i = 0;while (buffer[i]){start[i] = buffer[i];++i;}//start = buffer; 错误写法,因为这样写直接就将start指针修改,start就不是指向共享内存的地址sleep(10);//将共享内存段与当前进程脱离(shmdt)int n = shmdt(start);assert(n != -1);(void)n;//4.删除共享内存(shmctl)int m = shmctl(shmid, IPC_RMID, nullptr); // IPC_RMID | 删除共享内存段assert(m != -1);(void)m;return 0;
}

client.cc代码如下:

#include "comm.hpp"int main()
{//1.获取已经存在的共享内存key_t k = getKey();int shmid = shmget(k, gsize, IPC_CREAT); //int shmget(key_t key, size_t size, int shmflg)// 单独使用IPC_CREAT: 创建一个共享内存,如果共享内存不存在,就创建之,如果已经存在,获取已经存在的共享内存并返回;// 这里获取的是已经存在的//2.将共享内存段连接到进程地址空间(shmat)char* start = (char*)shmat(shmid, nullptr, 0);if(*start == -1){cerr << "error: " << errno << " : " << strerror(errno) << endl;exit(3);}//3.从共享内存中读取数据int m = 3;while (m--){cout << "i am client,i read: " << start << endl;sleep(3);}//将共享内存段与当前进程脱离(shmdt)int n = shmdt(start);assert(n != -1);(void)n;return 0;
}

makefile代码如下:

.PHONY:all
all:server clientserver:server.ccg++ -o $@ $^ -std=c++11
client:client.ccg++ -o $@ $^ -std=c++11.PHONY:clean
clean:rm -f server client

运行结果如下:
在这里插入图片描述

如果在自写代码中有如下错误,File exists,这是因为执行server.cc程序,程序并不是完整退出,而是程序进行一半时退出,例如:程序进行一半时按ctrl+c强制退出,程序没有执行到最后,也就是共享内存没有被删除,这时,就可以用ipcs -m查看共享内存;
在这里插入图片描述
然后ipcrm -m shmid(4)如下shmid为4,进行删除,重新运行程序即可。
在这里插入图片描述
如果是别的问题,那一定是代码错着,仔细检查代码吧。

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

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

相关文章

跨境电商中的安全利器与效率助推器

在如今全球互联网时代&#xff0c;跨境电商已成为全球贸易的重要组成部分。然而&#xff0c;对于从事跨境电商的企业和个人而言&#xff0c;面对诸多网络隐私与安全风险&#xff0c;如何确保在线操作的安全性和高效性成为了摆在他们面前的重要课题。本文将向您介绍隐擎Fox指纹浏…

从Spring 应用上下文获取 Bean

ApplicationContext 提供了获取所有已经成功注入 Spring IoC 容器的 Bean 名称的方法 getBeanDefinitionNames() 。然后我们可以借助于其 getBean(String name) 方法使用 Bean 名称获取特定的 Bean。 我们使用 CommandLineRunner 接口来打印一下结果。 1.1 获取所有的 Bean im…

手机APP智能硬件开发,蓝牙设备如何测试?

一、APP扫描设备 1、手机端蓝牙不同设置下是否能正常扫描&#xff1a;蓝牙打开、蓝牙关闭&#xff1b; 2、蓝牙设备被扫描到并出现在可连接设备列表的条件&#xff1a; 蓝牙设备名称与可连接的设备列表中的名称匹配、设备处于广播状态&#xff1b; 蓝牙设备与可连接的设备列…

【前端技巧】CSS常用知识碎片(九)

CSS常用知识碎片&#xff08;九&#xff09; mask-image属性 带有半透明的PNG图像的遮罩效果 .mask-image {mask: no-repeat center / contain;mask-image: url(bird.png); }SVG图形遮罩效果 .mask-image {mask-image: url("data:image/svgxml,%3Csvg viewBox0 0 3232…

mysql 字符 1024个字符限制 cast转为varchar 不限制字符长度 最大字符长度 group_concat长度限制

设置group_concat的最大长度然后再运行 SET SESSION group_concat_max_len102400;

【算法基础:数据结构】2.2 字典树/前缀树 Trie

文章目录 知识点cpp结构体模板 模板例题835. Trie字符串统计❤️❤️❤️❤️❤️&#xff08;重要&#xff01;模板&#xff01;&#xff09;143. 最大异或对&#x1f62d;&#x1f62d;&#x1f62d;&#x1f62d;&#x1f62d;&#xff08;Trie树的应用&#xff09; 相关题目…

C# MVC 多图片上传预览

一.效果图&#xff1a; 开发框架&#xff1a;MVC&#xff0c;Layui 列表主界面这里就不展示了&#xff0c;可以去看看这篇文章&#xff1a;Layui项目实战&#xff0c;这里讲的是“上传Banner”界面功能&#xff1a; 其中包括&#xff0c;多文件上传&#xff0c;预览&#xff0c…

vue进阶-消息的订阅与发布

&#x1f4d6;vue基础学习-组件 介绍了嵌套组件间父子组件通过 props 属性进行传参。子组件传递数据给父组件通过 $emit() 返回自定义事件&#xff0c;父组件调用自定义事件接收子组件返回参数。 &#x1f4d6;vue进阶-vue-route 介绍了路由组件传参&#xff0c;两种方式&…

【conan】本地编译三方库,上传conan服务器

1.6 conan 远程已经编译好的库 conan中文博客&#xff1a; 三方库资源&#xff1a; github conan-io 本地查询 conan search Existing package recipes:b2/4.9.6 boost/1.71.0nolovr/stable bzip2/1.0.8 ceres-solver/2.0.0nolovr/stable eigen/3.3.7nolovr/stable eigen_c…

【软件测试】selenium中元素的定位

1.元素的定位 不管用那种方式&#xff0c;必须保证页面上该属性的唯一性 1.CSS 定位 CSS(Cascading Style Sheets)是一种语言&#xff0c;它被用来描述HTML 和XML 文档的表现。 CSS 使用选择器来为页面元素绑定属性。这些选择器可以被selenium 用作另外的定位策略CSS的获取可…

Go 工具链详解(三): 代码测试神器 go test

go test 作用 go test 是 Go 工具链中的一个命令&#xff0c;用于编译和运行按照要求编写的 Golang 测试代码&#xff0c;并生成测试报告。 要求将测试代码所在的文件命名为 *_test.go&#xff0c;如此命名的文件不会被 go build 命令编译&#xff0c;但是会被 go test 进行编…

C++基础算法前缀和和差分篇

&#x1f4df;作者主页&#xff1a;慢热的陕西人 &#x1f334;专栏链接&#xff1a;C算法 &#x1f4e3;欢迎各位大佬&#x1f44d;点赞&#x1f525;关注&#x1f693;收藏&#xff0c;&#x1f349;留言 主要讲解了前缀和和差分算法 文章目录 Ⅳ. 前缀和 和 差分Ⅵ .Ⅰ前缀和…

echarts_柱状图+漏斗图

目录 柱状图(bar)需求[1] 复制案例[2] 修改类目轴方向[3] 修改数据渲染方向[4] 修改坐标轴文本样式 漏斗图(funnel)漏斗图的形状 柱状图(bar) 需求 如上图&#xff0c;做一个横向柱状图&#xff0c;后端返回的数据是从小向大排列的数据&#xff0c;希望能够按照顺序进行展示。…

【Docker】详解docker安装及使用

详解docker安装及使用 1. 安装docker1.1 查看docker版本信息 2. Docker镜像操作3. Docker容器操作4.知识点总结4.1 docker镜像操作4.2 docker容器操作4.3 docker run启动过程 参见docker基础知识点详解 1. 安装docker 目前Docker只能支持64位系统。 ###关闭和禁止防火墙开机自…

pytorch+CRNN实现

最近接触了一个仪表盘识别的项目&#xff0c;简单调研以后发现可以用CRNN来做。但是手边缺少仪表盘数据集&#xff0c;就先用ICDAR2013试了一下。 结果遇到了一系列坑。为了不使读者和自己在以后的日子继续遭罪。我把正确的代码发到下面了。 1&#xff09;超参数请不要调整&am…

Android oom_adj 详细解读

源码基于&#xff1a;Android R 0. 前言 在博文《oom_adj 内存水位算法剖析》一文中详细的分析了lmkd 中针对 oom_adj 内存水位的计算、使用方法&#xff0c;在博文《oom_adj 更新原理(1)》、《oom_adj 更新原理(2)》中对Android 系统中 oom_adj 的更新原理进行了详细的剖析。…

Linux Shell脚本文件

文章目录 Linux Shell脚本文件vim编辑器vi的使用 认识Bash变量环境变量用户操作变量delcare与typeset 命令的别名与历史命名 正则表达式Shell ScriptsShell Scripts良好习惯第一个shell其他shell例子 - 来自鸟哥linux私房菜语法介绍shell 的调试 Linux Shell脚本文件 vim编辑器…

Centos 7 安装 Oracle 11G

Oracle 11G 安装教程 准备环境 p13390677_112040_Linux-x86-64_1of7.zipp13390677_112040_Linux-x86-64_2of7.zipCentos 7- rhel7-英文版的系统–不想换语言的执行(LANGen_US)– 传输 文件到服务器上 创建用户和组 [rootlocalhost ~]# groupadd oracle [rootlocalhost ~]…

20230715----重返学习-vue3新API-Vue3和Vue2对比-vue3语法-Vue3编码

day-113-one-hundred-and-thirteen-20230715-vue3新API-Vue3和Vue2对比-vue3语法-Vue3编码 vue3新API Vue 3 迁移指南 虚拟DOM 说明:看01视频。虚拟DOM是用于让vue核心代码脱离浏览器的限制&#xff0c;在微信小程序、手机端、canvas之类也有能使用vue语法的可能&#xff0…

Windows11 C盘瘦身

1.符号链接 将大文件夹移动到其他盘&#xff0c;创建成符号链接 2.修改Android Studio路径设置 1.SDK路径 2.Gradle路径 3.模拟器路径 设置环境变量 ANDROID_SDK_HOME