Linux进程间通信(IPC)机制之一:共享内存

                                               🎬慕斯主页修仙—别有洞天

                                              ♈️今日夜电波:Nonsense—Sabrina Carpenter

                                                                0:50━━━━━━️💟──────── 2:43
                                                                    🔄   ◀️   ⏸   ▶️    ☰  

                                      💗关注👍点赞🙌收藏您的每一次鼓励都是对我莫大的支持😍


目录

什么是共享内存?

共享内存介绍

共享内存原理

函数接口详解

通过ftok获取key值

通过shmget创建共享内存

一些小细节

通过shmat挂接进程

通过shmdt取消与共享内存的关联

通过shmctl控制共享内存

IPC_RMID:删除共享内存

IPC_STAT:获取共享内存的状态

IPC_SET:改变共享内存的状态

共享内存的拓展

Makefile

server.cc

client.cc

代码效果


什么是共享内存?

共享内存介绍

        Linux共享内存是一种快速的数据交换手段,允许多个进程访问同一块内存区域

        共享内存在Linux中是一种高效的进程间通信(IPC)机制,它使得不同的进程可以访问同一段内存区域,从而实现数据共享和传输。它是内核级别的资源,并且通常是进程间通信方式中最快的一种。Linux共享内存的使用方式主要有以下几种(本文主要介绍基于system V的共享内存):

  1. 基于system V的共享内存:这是传统的方法,历史悠久,但API较为复杂。如果编译内核时没有选择CONFIG_SYSVIPC,则不会支持这种方式。
  2. 基于POSIX mmap文件映射实现共享内存:这种方法使用mmap系统调用将文件映射到内存中,从而实现共享。
  3. 通过memfd_create()和fd跨进程共享:这是一种较新的方法,用于创建匿名内存区域,并通过文件描述符在不同进程间共享。
  4. 基于dma-buf的共享内存:这在多媒体和图形领域广泛使用,适用于高性能的数据传输需求。

        共享内存的优势在于其高速的数据传输能力,因为数据不需要在用户空间和内核空间之间复制。然而,它也有一些劣势,比如需要手动管理同步和并发访问,以及可能的安全问题,因为它绕过了操作系统的正常内存保护机制。为了提高安全性,可以使用命名管道等机制来实现访问控制。

共享内存原理

        我们还是需要遵守一句话:进程间通信的前提是让不同的进程看到同一块资源(必须由OS提供)。共享内存区是最快的IPC形式。一旦这样的内存映射到共享它的进程的地址空间,这些进程间数据传递不再涉及到内核,换句话说是进程不再通过执行进入内核的系统调用来传递彼此的数据。也就是说他是在用户空间中的而不是在内核空间中(缓冲区、文件属性等等就是在此)。如下是简易的图示:

​        更为详细的图示:

        共享内存的原里实际上就是(1)在物理内存中申请一块空间(2)将申请的空间通过页表映射到进程的虚拟内存中的共享区中,共享区通过返回虚拟地址的首地址就可让进程看到资源。接着再让另外的进程做同样的操作,不够需要注意的是要指向同一块物理空间。这样我们就让不同的进程看到了同一块资源。(3)当我们去掉页表的映射关系。(4)释放物理内存的空间后,就解除了共享内存。

        需要注意的是:系统中一定会有很多的共享内存存在,操作系统需要管理全部共享内存,因此我们会按照“先描述,在组织”的原则创建对应的数据结构进行管理。共享内存会必须要求有唯一标识符来区分,我们需要制定一定的规则、约定让不同的进程识别同一块共享内存从而得以通信

函数接口详解

        共享内存函数Linux共享内存的相关函数主要包括shmgetshmatshmdtshmctl。具体如下:

         1、shmget:这是创建或获取共享内存段的函数。它的原型是

#include <sys/ipc.h>
#include <sys/shm.h>
int shmget(key_t key, size_t size, int shmflg);
    • key:用于指定共享内存的键值,可以是任意非负整数,通常通过ftok函数生成。
    • size:指定共享内存段的大小。
    • shmflg:设置共享内存段的权限和属性。

         2、shmat:这个函数用于将共享内存段附加到进程的地址空间上。它的原型是

#include <sys/types.h>
#include <sys/shm.h>
void *shmat(int shmid, const void *shmaddr, int shmflg);
    • shmid:由shmget返回的共享内存标识符。
    • shmaddr:可选参数,通常设置为nullptr,让系统自动选择一个地址来附加共享内存。
    • shmflg:附加标志,如SHM_RDONLY以只读方式附加。

         3、shmdt:当进程完成对共享内存的使用后,需要使用shmdt函数将其从进程的地址空间中分离(detach)。它的原型是

#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/shm.h>
int shmdt(void *addr);
    • addr:之前通过shmat附加的共享内存地址。

         4、shmctl:这个函数用于对共享内存段进行控制操作,如删除共享内存段。它的原型是:

#include <sys/ipc.h>
#include <sys/shm.h>
int shmctl(int shmid, int cmd, struct shmid_ds *buf);
  • shmid:共享内存标识符,即要控制的共享内存段的标识符。
  • cmd:控制命令,可取值如下:
    • IPC_STAT:获取共享内存的状态。
    • IPC_SET:改变共享内存的状态。
    • IPC_RMID:删除共享内存。
  • buf:指向struct shmid_ds结构的指针,该结构用于存储共享内存的状态信息。当cmd为IPC_STAT或IPC_SET时,需要使用此参数。

        在实际应用中,通常会结合信号量或互斥锁等同步机制来确保数据的一致性和访问的安全性。此外,共享内存的使用也需要考虑到系统的资源限制,例如共享内存段的大小和数量都可能受到系统配置的限制。

通过ftok获取key值

        如果我们要找到或者创建一个共享内存,我们需要约定一个相同的key值,这样才能找到对应的共享内存,才能让进程间相互通信。因此,这也是创建共享内纯的前提。ftok函数用于将一个已存在的路径名和一个整数标识符转换成IPC键值,以便在进程间通信中使用

    ftok函数是Linux系统编程中用于创建唯一键值的工具,这个键值通常用于进程间通信(IPC)的机制,如消息队列、信号量和共享内存。它的函数原型如下:

#include <sys/types.h>
#include <sys/ipc.h>
key_t ftok(const char *pathname, int proj_id);
  • 参数说明
    • pathname:这是一个已存在的文件路径名,ftok会从这个路径名导出信息。
    • proj_id:这是一个与路径名相关的项目ID,通常是0到255之间的整数。
  • 返回值:函数返回一个key_t类型的键值,这个键值是由pathname导出的信息与proj_id的低序8位组合而成的。

        在使用ftok时,需要注意以下几点:

  • 确保提供的pathname确实存在,因为ftok会根据这个路径名生成键值的一部分。
  • proj_id应该是一个相对较小的整数,因为它只会使用其低序8位。
  • 生成的键值应该是唯一的,以确保不同的进程间通信不会相互干扰。

        如下示例:

#include <iostream>
#include <cstdlib>
#include <string>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <cstring>const std::string pathname = "/home/land/109/pip/systemV";
const int proj_id = 0x11223344;key_t GetKey()
{key_t key = ftok(pathname.c_str(), proj_id);if (key < 0){std::cerr << "errno: " << errno << ", errstring: " << strerror(errno) << std::endl;exit(1);}std::cout <<"key:"<< key << std::endl;return key;
}

通过shmget创建共享内存

        通过手册理解:

​        我们最经常使用的是IPC_CREAT、IP_EXCL。他们分别的含义是:

IPC_CREAT//shm不存在,就创建。存在,就获取并返回。
IP_EXCL//不能单独使用,shm不存在就创建,存在就出错返回。

        如下示例:

    key_t key = GetKey();int shmid = shmget(key, size, IPC_CREAT | IPC_EXCL);if (shmid < 0){std::cerr << "errno: " << errno << ", errstring: " << strerror(errno) << std::endl;exit(1);}std::cout << "shmid:" << shmid << std::endl;return 0;

​        穿插认识一下查看和删除共享内存的命令:

ipcs -m//显示共享内存的属性
ipcrm -m shmid//删除指定的共享内存

一些小细节

        前面我们在shmget的shmflg的介绍中知道这实际上还包含权限的设置,实际上要设置权限只需要在选完IPC_CREAT或IP_EXCL等后 | 上对应权限即可,如下:

    key_t key = GetKey();int shmid = shmget(key, size, IPC_CREAT | IPC_EXCL | 0666);if (shmid < 0){std::cerr << "errno: " << errno << ", errstring: " << strerror(errno) << std::endl;exit(1);}std::cout << "shmid:" << shmid << std::endl;return 0;

        当我们设置要申请的共享内存大小时,强烈建议申请4096的整数倍数大的大小,因为低层分配是按4096的整数倍数来分配大小的,比如:你申请4097大小,但是低层实际上是4096*2的大小。

通过shmat挂接进程

        通过shmaddr可以选择一个地址来附加共享内存,但是通常我们会填nullptr,shmflg填是0值得注意的是:它的返回值是一个void* 的挂接成功后的虚拟地址空间的起始地址,也就是之前原理所述start_addr。如下是对应的手册:

        如下示例:

int main()
{key_t key = GetKey();int shmid = shmget(key, size, IPC_CREAT | IPC_EXCL | 0666);if (shmid < 0){std::cerr << "errno: " << errno << ", errstring: " << strerror(errno) << std::endl;exit(1);}sleep(5);std::cout << "shmid:" << shmid << std::endl;std::cout<<"开始将shm映射到进程的地址空间中"<<std::endl;char* s=(char*)shmat(shmid,nullptr,0);sleep(10);return 0;
}

通过shmdt取消与共享内存的关联

        根据上shmat的返回值,也就是虚拟地址空间的起始地址来实现取消与共享内存的关联。实际上,取消关联的本质就是修改页表,我们可以从起始地址连续释放对应的虚拟内存的大小,在页表中连续释放对应大小的映射关系。根据shmat的示例来取消关联:

int main()
{key_t key = GetKey();int shmid = shmget(key, size, IPC_CREAT | IPC_EXCL | 0666);if (shmid < 0){std::cerr << "errno: " << errno << ", errstring: " << strerror(errno) << std::endl;exit(1);}sleep(5);std::cout << "shmid:" << shmid << std::endl;std::cout<<"开始将shm映射到进程的地址空间中"<<std::endl;char* s=(char*)shmat(shmid,nullptr,0);sleep(10);shmdt((char*)s);sleep(5);return 0;
}

通过shmctl控制共享内存

IPC_RMID:删除共享内存

        如下示例:

// 定义共享内存结构体变量,当然也可直接传nullptr
struct shmid_ds shmbuffer;
// 获取共享内存状态
int ret = shmctl(shmid, IPC_STAT, &shmbuffer);
if (ret == -1) {perror("shmctl");exit(1);
}

IPC_STAT:获取共享内存的状态

        如下示例:

#include <sys/ipc.h>
#include <sys/shm.h>
#include <stdio.h>int main() {key_t key = GetKey();// 共享内存标识符int shmid = shmget(key, size, IPC_CREAT | IPC_EXCL | 0664);// 定义共享内存状态结构体变量struct shmid_ds shmbuffer;// 获取共享内存状态int ret = shmctl(shmid, IPC_STAT, &shmbuffer);if (ret == -1) {perror("shmctl");return 1;}// 打印共享内存状态信息printf("共享内存状态信息:\n");printf("当前连接数: %d\n", shmbuffer.shm_nattch);printf("最后一次操作进程ID: %d\n", shmbuffer.shm_lpid);printf("最后一次操作时间: %ld\n", shmbuffer.shm_change_time);printf("共享内存大小: %ld\n", shmbuffer.shm_segsz);return 0;
}

IPC_SET:改变共享内存的状态

        如下示例:

#include <sys/ipc.h>
#include <sys/shm.h>
#include <stdio.h>int main() {key_t key = GetKey();// 共享内存标识符int shmid = shmget(key, size, IPC_CREAT | IPC_EXCL | 0664);// 定义共享内存状态结构体变量struct shmid_ds shmbuffer;// 获取共享内存状态int ret = shmctl(shmid, IPC_STAT, &shmbuffer);if (ret == -1) {perror("shmctl");return 1;}// 修改共享内存状态shmbuffer.shm_perm.mode |= 0666; // 设置共享内存权限为可读写shmbuffer.shm_perm.uid = getuid(); // 设置共享内存所有者为用户IDshmbuffer.shm_perm.gid = getgid(); // 设置共享内存所属组为用户组ID// 更新共享内存状态ret = shmctl(shmid, IPC_SET, &shmbuffer);if (ret == -1) {perror("shmctl");return 1;}printf("共享内存状态已成功更新。\n");return 0;
}

共享内存的拓展

        值得注意的是:共享内存是没有同步机制的,但是可以通过其他方法实现同步。由于多个进程可能同时读写共享内存,因此需要一种同步机制来确保数据的一致性和防止竞态条件的发生。以下是一些常用的同步方法:

  • 互斥锁(Mutexes):互斥锁是一种用于保护共享资源不被多个线程同时访问的同步机制。在Linux中,可以使用pthread库中的互斥锁来实现进程间的同步。
  • 信号量(Semaphores):信号量是另一种用于同步不同进程或线程的机制。它可以控制对共享资源的访问数量。在Linux中,可以使用POSIX有名信号量或POSIX基于内存的信号量来实现同步。
  • 信号(Signals):虽然信号主要用于进程间的通知,但它们也可以用于同步。例如,一个进程可以发送信号给另一个进程,告知它共享内存已经更新,从而触发接收进程执行某些操作。

        需要注意的是:在使用这些同步机制时,应当小心避免死锁和活锁的情况。此外,设计良好的同步策略对于提高系统性能和可靠性至关重要。

        下面是一个通过命名管道控制共享内存同步的例子:通过命名管道控制client每秒向共享内存写入一个字母,每次写入成功后会发送一个信号给server,server在收到信号后才去读取共享内存中的内容,并且将读取到的内容打印出来,如下为完整的代码以及效果:

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 fifo

server.cc

#include <iostream>
#include <cstring>
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/msg.h>
#include <sys/shm.h>
#include <sys/sem.h>
#include <unistd.h>
#include "comm.hpp"class Init
{
public:Init(){bool r = MakeFifo();if (!r)return;key_t key = GetKey();std::cout << "key : " << ToHex(key) << std::endl;shmid = CreateShm(key);std::cout << "shmid: " << shmid << std::endl;std::cout << "开始将shm映射到进程的地址空间中" << std::endl;s = (char *)shmat(shmid, nullptr, 0);fd = open(filename.c_str(), O_RDONLY);}~Init(){shmdt(s);std::cout << "开始将shm从进程的地址空间中移除" << std::endl;shmctl(shmid, IPC_RMID, nullptr);std::cout << "开始将shm从OS中删除" << std::endl;close(fd);unlink(filename.c_str());}public:int shmid;int fd;char *s;
};int main()
{Init init;//TODOwhile (true){// waitint code = 0;ssize_t n = read(init.fd, &code, sizeof(code));if (n > 0){// 直接读取std::cout << "共享内存的内容: " << init.s << std::endl;sleep(1);}else if (n == 0){break;}}sleep(10);return 0;
}

client.cc

#include <iostream>
#include <cstring>
#include <unistd.h>
#include <sys/ipc.h> //Inter-Process Communication
#include <sys/shm.h>
#include "comm.hpp"int main()
{key_t key = GetKey();int shmid = GetShm(key);char *s = (char*)shmat(shmid, nullptr, 0);std::cout << "attach shm done" << std::endl;int fd = open(filename.c_str(), O_WRONLY);// sleep(10);// TODO// 共享内存的通信方式,不会提供同步机制, 共享内存是直接裸露给所有的使用者的,一定要注意共享内存的使用安全问题// char c = 'a';for(; c <= 'z'; c++){s[c-'a'] = c;std::cout << "write : " << c << " done" << std::endl;sleep(1);// 通知对方int code = 1;write(fd, &code, sizeof(4));}shmdt(s);std::cout << "detach shm done" << std::endl;close(fd);return 0;
}

代码效果


                   感谢你耐心的看到这里ღ( ´・ᴗ・` )比心,如有哪里有错误请踢一脚作者o(╥﹏╥)o! 

                                       

                                                                        给个三连再走嘛~  

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

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

相关文章

【Linux C | 网络编程】getsockname 和 getpeername函数详解及C语言例子

&#x1f601;博客主页&#x1f601;&#xff1a;&#x1f680;https://blog.csdn.net/wkd_007&#x1f680; &#x1f911;博客内容&#x1f911;&#xff1a;&#x1f36d;嵌入式开发、Linux、C语言、C、数据结构、音视频&#x1f36d; &#x1f923;本文内容&#x1f923;&a…

Opencv——霍夫变换

霍夫直线变换 霍夫直线变换(Hough Line Transform)用来做直线检测 为了加升大家对霍夫直线的理解,我在左图左上角大了一个点,然后在右图中绘制出来经过这点可能的所有直线 绘制经过某点的所有直线的示例代码如下,这个代码可以直接拷贝运行 import cv2 as cv import matplot…

Android systemui 编译

目录 简介&#xff1a; 一、步骤 二、下载源码 三、环境配置 四、确定好需要编译版本 五、编译SystemUI 步骤1&#xff1a;进入源代码目录 步骤2&#xff1a;初始化编译环境 步骤3&#xff1a;选择目标设备 步骤4&#xff1a;编译SystemUI 步骤5&#xff1a;查找生成…

Unity3D正则表达式的使用

系列文章目录 unity工具 文章目录 系列文章目录前言一、匹配正整数的使用方法1-1、代码如下1-2、结果如下 二、匹配大写字母2-1、代码如下1-2、结果如下 三、Regex类3-1、Match&#xff08;&#xff09;3-2、Matches()3-3、IsMatch&#xff08;&#xff09; 四、定义正则表达式…

ModelArts加速识别,助力新零售电商业务功能的实现

前言 如果说为客户提供最好的商品是产品眼中零售的本质&#xff0c;那么用户的思维是什么呢&#xff1f; 在用户眼中&#xff0c;极致的服务体验与优质的商品同等重要。 企业想要满足上面两项服务&#xff0c;关键在于提升效率&#xff0c;也就是需要有更高效率的零售&#…

ISCTF wp

web 圣杯战争 题目源码 <?php highlight_file(__FILE__); error_reporting(0);class artifact{public $excalibuer;public $arrow;public function __toString(){echo "为Saber选择了对的武器!<br>";return $this->excalibuer->arrow;} }class pre…

vue预览pdf文件的几种方法

文章目录 vue预览pdf集中方法方法一&#xff1a;方法二&#xff1a;展示效果&#xff1a;需要包依赖&#xff1a;代码&#xff1a; 方法三&#xff1a;展示效果&#xff1a;需要包依赖&#xff1a;代码&#xff1a;自己调参数&#xff0c;选择符合自己的 vue预览pdf集中方法 我…

WordPress主题YIA导航菜单中如何添加Iconfont字体图标?

YIA主题自带的字体图标比较少&#xff0c;一般用于文章属性或分享、点赞等地方&#xff0c;而导航菜单中的字体图标可以说没有&#xff0c;所以想要在菜单中添加字体图标的话&#xff0c;只能自行添加了。下面boke112百科就跟大家详细说一说WordPress主题YIA导航菜单中添加Icon…

【算法】登山(线性DP,最长上升)

题目 五一到了&#xff0c;ACM队组织大家去登山观光&#xff0c;队员们发现山上一共有N个景点&#xff0c;并且决定按照顺序来浏览这些景点&#xff0c;即每次所浏览景点的编号都要大于前一个浏览景点的编号。 同时队员们还有另一个登山习惯&#xff0c;就是不连续浏览海拔相…

单元/集成测试服务

服务概述 单元/集成测试旨在证明被测软件实现其单元/架构设计规范、证明被测软件不包含非预期功能。经纬恒润测试团队拥有丰富的研发经验、严格的流程管控&#xff0c;依据ISO26262/ASPICE等开展符合要求的单元测试/集成测试工作。 在ISO 26262 - part6 部分产品开发&#xff…

Android Studio 安装配置教程 - Windows版

Android Studio下载 安装&#xff1a; 下载&#xff1a; Android Studio Hedgehog | 2023.1.1 | Android Developers (google.cn) 安装&#xff1a; 基本不需要思考跟着走 默认下一步 默认下一步 自定义修改路径&#xff0c;下一步 默认下一步&#xff0c;不勾选 默认下一…

状态码400以及状态码415

首先检查前端传递的参数是放在header里边还是放在body里边。 此图前端传参post请求&#xff0c;定义为’Content-Type’&#xff1a;‘application/x-www-form-urlencoded’ 此刻他的参数在FormData中。看下图 后端接参数应为&#xff08;此刻参数前边什么都不加默认为requestP…

GitHub工作流的使用笔记

文章目录 前言1. 怎么用2. 怎么写前端案例1&#xff1a;自动打包到新分支前端案例2&#xff1a;自动打包推送到gitee的build分支案例3&#xff1a;暂时略 前言 有些东西真的就是要不断的试错不断地试错才能摸索到一点点&#xff0c;就是摸索到凌晨两三点第二天要8点起床感觉要…

Kotlin 协程1:深入理解withContext

Kotlin 协程1&#xff1a;深入理解withContext 引言 在现代编程中&#xff0c;异步编程已经变得非常重要。在 Kotlin 中&#xff0c;协程提供了一种优雅和高效的方式来处理异步编程和并发。在这篇文章中&#xff0c;我们将深入探讨 Kotlin 协程中的一个重要函数&#xff1a;wi…

Java基础—面向对象—19static关键字详解、抽象类、接口、N种内部类

1、static关键字 匿名代码块、静态代码块、构造方法 静态代码块是在类加载的时候执行&#xff0c;仅执行一次 匿名代码块在调用构造函数之前 验证如下图&#xff1a; 2、静态导入包&#xff08;可能很多人听都没听过&#xff09; 3、Math是用final关键字的&#xff0c;fina…

【操作系统·考研】目录

1.概述 与文件管理系统和文件集合相关联的是文件目录&#xff0c;它包含有关文件的属性、位置和所有权等。 2.目录的结构 2.1 单级目录结构 在整个FS中只建立一张目录表&#xff0c;每个文件占一个目录项。 当访问一个文件时&#xff0c;先根据文件名在目录表中找到相应的FC…

数据结构day7

1.思维导图 1.二叉树递归创建 2.二叉树先中后序遍历 3.二叉树计算节点 4.二叉树计算深度。 5.编程实现快速排序降序

Ubuntu-22.04上ToDest设置开机不弹出图形界面

提示&#xff1a;文章写完后&#xff0c;目录可以自动生成&#xff0c;如何生成可参考右边的帮助文档 文章目录 前言一、开始操作1.设置图形端 总结 前言 有时候远程成为开发必不可少的工具&#xff0c;目前国内有很多相关的软件&#xff0c;比较有名的是向日葵、ToDesk、Rust…

Hutool导入导出用法

整理了下Hutool导入导出的简单使用。 导入maven或jar包&#xff08;注意这里导入的poi只是为了优化样式&#xff09; <!-- https://mvnrepository.com/artifact/cn.hutool/hutool-all --> <dependency><groupId>cn.hutool</groupId><artifactId&g…

iOS17使用safari调试wkwebview

isInspectable配置 之前开发wkwebview的页面的时候一直使用safari调试&#xff0c;毕竟jssdk交互还是要用这个比较方便&#xff0c;虽说用一个脚本插件没问题。不过还是不太方便。 但是这个功能突然到了iOS17之后发现不能用了&#xff0c;还以为又是苹果搞得bug&#xff0c;每…