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.hqwc.cn/news/442548.html

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

相关文章

spring-boot-admin的介绍和使用

概述 Spring Boot 有一个非常好用的监控和管理的源软件&#xff0c;这个软件就是 Spring Boot Admin。该软件能够将 Actuator 中的信息进行界面化的展示&#xff0c;也可以监控所有 Spring Boot 应用的健康状况&#xff0c;提供实时警报功能。 主要的功能点有&#xff1a; 显…

chrome浏览器如何某网址设置禁止http自动跳转https

场景&#xff1a;网址 http://www.xxx.com 粘贴到浏览器会自动跳转为 https://www.xxx.com&#xff0c;但是仅是想设置该网址禁止制动跳转https。 操作&#xff1a; 1、chrome浏览器输入网址&#xff1a;chrome://net-internals/#hsts 2、在“Delete domain security polici…

腾讯云0基础10秒搭建幻兽帕鲁游戏联机服务器

幻兽帕鲁&#xff08;Palworld&#xff09;是一款多人在线游戏&#xff0c;为了获得更好的游戏体验&#xff0c;需要搭建一个稳定、高效的游戏联机服务器。腾讯云提供了一种简单、快速的方法&#xff0c;让新手小白也能0基础10秒搭建幻兽帕鲁游戏联机服务器&#xff01; 本文将…

pdf怎么转成高清图?pdf在线转换器推荐分享

在日常的工作或者学习中&#xff0c;有时候会需要将编辑好的pdf转高清图片&#xff0c;这样更方便我们后续使用&#xff0c;那么怎么将pdf转图片&#xff08;https://www.yasuotu.com/pdftopic&#xff09;还能保持清晰呢&#xff1f;下面介绍一款pdf转换工具&#xff0c;支持p…

【EI会议征稿通知】第五届能源电力与自动化工程国际学术会议(ICEPAE 2024)

第五届能源电力与自动化工程国际学术会议&#xff08;ICEPAE 2024&#xff09; 2024 5th International Conference on Energy Power and Automation Engineering 第五届能源电力与自动化工程国际学术会议&#xff08;ICEPAE 2024&#xff09;将于2024年5月24日至26日在中国 …

PMP备考笔记:模拟考试知识点总结

1. 答题思路&#xff1a;优先看问题&#xff0c;可节省时间。 2. 考试就按照考试的套路来做&#xff0c;不要过多考虑。 开发团队只专注当前冲刺目标&#xff0c;产品负责人对PB排优先级。 收集需求工具-原型法&#xff1a;能够让用户提前体验&#xff0c;减少返工的风险。 …

Unity | 资源热更(YooAsset AB)

目录 一、AssetBundle 1. 插件AssetBundle Browser 打AB包 &#xff08;1&#xff09;Unity&#xff08;我用的版本是2020.3.8&#xff09;导入AssetBundle Browser &#xff08;2&#xff09;设置Prefab &#xff08;3&#xff09;AssetBundleBrowser面板 2. 代码打AB包…

分布式搜索引擎_学习笔记_1

分布式搜索引擎01 – elasticsearch基础 0.学习目标 1.初识elasticsearch 1.1.了解ES 1.1.1.elasticsearch的作用 elasticsearch是一款非常强大的开源搜索引擎&#xff0c;具备非常多强大功能&#xff0c;可以帮助我们从海量数据中快速找到需要的内容 例如&#xff1a; …

Visual Studio无法调试Unity的可能原因和解决办法

问题描述&#xff1a; 在unity和vs都安装了相关插件的情况下&#xff0c;vs在启动了“附加到Unity”后却并没有进入调试模式。 可能的原因及解决办法&#xff1a; 1、Unity未设置成调试模式 将Unity编辑器的右下角这个debug标志设置成debug模式: 设置后变成了&#xff1a; 注…

mac英语学习工具:Eudic欧路词典v4.5.9增强激活版

Eudic欧路词典是一款功能强大的英语学习工具&#xff0c;其多语种支持、海量词库、强大的翻译功能、听力训练和生词本和笔记等特点&#xff0c;使得用户可以方便地进行英语学习和提高英语水平&#xff0c;适用于各种英语学习人员和文化交流人员等不同人群。 软件下载&#xff1…

字符串中的html标签解析

"bgghnhfvfgbghbgh<span>9996487878</span>jdkfjdklvqqqq/jdkfjdklvbgghnhfvfgbghbgh<span>9996487878</span>" 类似一个这样的字符串&#xff0c;i标签是通过正则把字符串jdkfjdklv和/jdkfjdklv替换而成的 &#xff0c; 想要把i标签的内容转…

电脑护眼模式怎么设置?4个有效方法保护眼睛!

“我感觉每天使用电脑的时间久了&#xff0c;眼睛总是不太舒服。电脑护眼模式怎么设置呢&#xff1f;有什么比较好用的方法可以推荐吗&#xff1f;” 如果长时间使用电脑&#xff0c;或许会让我们感到用眼疲劳。电脑护眼模式是现代人常用的电脑设置之一&#xff0c;它能有效地减…