当前位置: 首页 > article >正文

【Linux】<共享内存应用>——模拟实现不同进程把hello字符对<共享内存文件对象>的放入和取出操作

前言

大家好吖,欢迎来到 YY 滴Linux系列 ,热烈欢迎! 本章主要内容面向接触过C++ Linux的老铁
主要内容含:
在这里插入图片描述

欢迎订阅 YY滴C++专栏!更多干货持续更新!以下是传送门!

  • YY的《C++》专栏
  • YY的《C++11》专栏
  • YY的《Linux》专栏
  • YY的《数据结构》专栏
  • YY的《C语言基础》专栏
  • YY的《初学者易错点》专栏
  • YY的《小小知识点》专栏
  • YY的《单片机期末速过》专栏
  • YY的《C++期末速过》专栏
  • YY的《单片机》专栏
  • YY的《STM32》专栏
  • YY的《数据库》专栏
  • YY的《数据库原理》专栏

目录

  • 一.共享内存相关<前置知识>与相关指令
    • 1.共享内存
    • 2.共享内存文件所在目录
    • 3.查看共享内存文件属性
    • 4.查看共享文件内容
    • 5.以0覆盖共享文件
  • 二.共享内存相关函数
    • 1.shm_open() 函数用于创建或打开一个命名的共享内存对象
    • 2.ftruncate() 函数设置共享内存大小
    • 3.mmap()函数介绍
      • 【1】利用mmap()函数向共享内存写入数据
      • 【2】利用mmap()函数从共享内存访问&读取数据
    • 4.close() 函数用于关闭文件描述符&shm_unlink() 函数用于删除命名的共享内存对象
  • 三.模拟实现不同进程把hello字符对<共享内存文件对象>的放入和取出操作
    • 进程0:共享内存对象创建&初始化
    • 进程1:把hello字符往<共享内存文件对象>放入
    • 进程2:把hello字符从<共享内存文件对象>取出(模拟)
    • 进程3:删除进程0创建的共享文件/dev/shm/my_shared_memory

一.共享内存相关<前置知识>与相关指令

1.共享内存

  • 共享内存(Shared Memory)是一种允许多个进程访问同一块内存空间的机制。这种技术常用于进程间通信(IPC)和数据共享,因为它提供了一种高效的方式来传递大量数据,而无需通过内核进行数据的复制。

  • 在共享内存模型中,两个或多个进程可以 映射同一块物理内存到它们各自的地址空间 中。这意味着,当一个进程向这块内存写入数据时,其他进程可以立即看到这些更改,因为它们实际上是在访问 相同的内存位置

2.共享内存文件所在目录

/dev/shm

3.查看共享内存文件属性

stat /my_shared_memory

4.查看共享文件内容

hd /my_shared_memory

5.以0覆盖共享文件

dd /my_shared_memory

二.共享内存相关函数

1.shm_open() 函数用于创建或打开一个命名的共享内存对象

  • shm_open() 函数用于创建或打开一个命名的共享内存对象,并返回一个文件描述符,即 shm_fd。这个函数的原型如下:
int shm_open(const char *name, int oflag, mode_t mode);

name:共享内存对象的名称,这个名称必须以斜杠(/)开头,并且不能包含其他斜杠。例如,"/my_shared_memory" 是一个合法的名称。
oflag:打开标志,可以是 O_RDONLY(只读)、O_RDWR(读写)或 O_CREAT(如果对象不存在则创建它)等标志的组合。
mode:当 oflag 包含 O_CREAT 时,这个参数指定了新创建的共享内存对象的权限。 
     例如:0666:对象的目录权限。rw-rw-rw-====110 110 110

2.ftruncate() 函数设置共享内存大小

ftruncate(shm_fd, 4096); // 将共享内存大小设置为4KB

3.mmap()函数介绍

  • mmap()函数将一个文件或者其它对象映射进内存,文件被映射到多个页上
  • 通过mmap()映射后, 用户可以直接操作这段虚拟地址进行文件的读写等操作,而不必再调用read()、write()等系统调用
void* mmap(void* addr, size_t length, int prot, int flags, int fd, off_t offset);



addr:映射区的开始地址。通常设置为NULL,让系统自动选择地址
length:映射区的长度
prot:期望的内存保护标志,不能与文件的打开模式冲突。可以是以下值的组合:
     PROT_EXEC:页内容可以被执行。
     PROT_READ:页内容可以被读取。
     PROT_WRITE:页可以被写入。
     PROT_NONE:页不可访问。
flags:指定映射对象的类型、映射选项和映射页是否可以共享。
fd:有效的文件描述词。如果MAP_ANONYMOUS被设定,为了兼容问题,其值应为-1。
offset:被映射对象内容的起点。

【1】利用mmap()函数向共享内存写入数据

  • 使用 mmap() 函数将共享内存映射到进程的地址空间,然后通过指针操作来写入数据。例如:
int shm_fd;
char* ptr;

shm_fd = shm_open("/my_shared_memory", O_RDWR, 0666);//O_RDWR读写
ptr = mmap(NULL, 4096,  PROT_WRITE, MAP_SHARED, shm_fd, 0);//PROT_WRITE:页可以被写入

memcpy(ptr, "Hello, World!\0", 13);
//sprintf(ptr,"%s","Hello");

【2】利用mmap()函数从共享内存访问&读取数据

  • 使用 mmap() 函数将共享内存映射到进程的地址空间,然后通过指针操作来读取访问数据。例如:
int shm_fd;
char* ptr;

//使用printf访问数据
shm_fd = shm_open("/my_shared_memory", O_RDWR, 0666);//O_RDWR读写
ptr = mmap(NULL, 4096,  PROT_READ, MAP_SHARED, shm_fd, 0);//PROT_READ:页内容可以被读取
printf("%s\n",ptr);

// 使用read()函数读取数据
char *buf = (char *)malloc(64);
lseek(shm_fd, 0, SEEK_SET); // 将文件指针重新定位到文件开头
read(shm_fd, buf, 13);

4.close() 函数用于关闭文件描述符&shm_unlink() 函数用于删除命名的共享内存对象

  • 不再需要使用共享内存时,应该关闭 shm_fd 并释放相关资源。这可以通过调用 close() 函数和 shm_unlink() 函数来实现。
  • close() 函数用于关闭文件描述符,而 shm_unlink() 函数用于删除命名的共享内存对象 (注意,这并不会立即释放共享内存,直到所有访问该共享内存的进程都退出后,它才会被真正释放)
close(shm_fd); // 关闭文件描述符
shm_unlink("/my_shared_memory"); // 删除命名的共享内存对象

三.模拟实现不同进程把hello字符对<共享内存文件对象>的放入和取出操作

进程0:共享内存对象创建&初始化

include <sys/mman.h>
include <fcntl.h>
include <unistd.h>

void main(){

    int shm_fd;
    
    shm_fd = shm_open("/my_shared_memory", O_RDWR, 0666);//O_RDWR读写
    ftruncate(shm_fd, 4096); // 将共享内存大小设置为4KB
    
}

进程1:把hello字符往<共享内存文件对象>放入

include <sys/mman.h>
include <fcntl.h>
include <unistd.h>
void main(){

    int shm_fd;
    char* ptr;
    
    shm_fd = shm_open("/my_shared_memory", O_RDWR, 0666);//O_RDWR读写
    ptr = mmap(NULL, 4096,  PROT_WRITE, MAP_SHARED, shm_fd, 0);//PROT_WRITE:页可以被写入
 
    sprintf(ptr,"%s","Hello");

}

进程2:把hello字符从<共享内存文件对象>取出(模拟)

include <sys/mman.h>
include <fcntl.h>
include <unistd.h>
void main(){

    int shm_fd;
    char* ptr;

    //使用printf访问数据
    shm_fd = shm_open("/my_shared_memory", O_RDWR, 0666);//O_RDWR读写
    ptr = mmap(NULL, 4096,  PROT_READ, MAP_SHARED, shm_fd, 0);//PROT_READ:页内容可以被读取
    printf("%s\n",ptr);
    
}

进程3:删除进程0创建的共享文件/dev/shm/my_shared_memory

include <sys/mman.h>
include <fcntl.h>
include <unistd.h>
void main(){
    shm_unlink("/my_shared_memory"); // 删除命名的共享内存对象
}

http://www.kler.cn/a/404271.html

相关文章:

  • ts: 定义一个对象接收后端返回对象数据,但是报错了有红色的红线为什么
  • 问题: redis-高并发场景下如何保证缓存数据与数据库的最终一致性
  • 241121学习日志——[CSDIY] [InternStudio] 大模型训练营 [11]
  • 【Linux】<共享内存应用>——模拟实现不同进程把hello字符对<共享内存文件对象>的放入和取出操作
  • K8S资源限制之ResourceQuota
  • 从技术到产品:第三方美颜API助力实时直播平台的开发详解
  • Seatunnel运行时报错Caused by: java.lang.NoClassDefFoundError: com/mysql/cj/MysqlType
  • 2024年亚太地区数学建模大赛C题-宠物行业及相关行业的发展分析与策略
  • 如何查看 Android 项目的依赖结构树
  • 集成学习的起源与哲学
  • Qt添加外部库:静态库和动态库,批量添加头文件
  • 【XSS】payload#1
  • 利用oss进行数据库和网站图片备份
  • Flowable第三篇、Flowable之任务分配(任务分配、流程变量、候选人和候选人组)
  • 【Lucene】搜索引擎和文档相关性评分 BM25 算法的工作原理
  • 大数据新视界 -- 大数据大厂之 Impala 性能优化:资源分配与负载均衡的协同(下)(24 / 30)
  • 如何使用GPT API 自定义 自己的 RAG
  • [羊城杯 2020]easyre
  • Java基础夯实——2.6 Java中的锁
  • 【Linux网络编程】套接字使用--TCP echo server的实现
  • 【腾讯云产品最佳实践】腾讯云CVM入门技术与实践:通过腾讯云快速构建云上应用
  • Jmeter中的断言(三)
  • 《Vue零基础入门教程》第一课:Vue简介
  • 初识ArkUI
  • SAP BC 记录一次因为HANA服务器内存满的问题
  • 如何选择合适的数据集成工具或平台来实现全域数据的高效整合