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

共享内存-systemV

01. 共享内存简述

共享内存是一个允许多个进程直接访问同一块物理内存区域的进程通信工具,因其本身不涉及用户态与核心态之间转换,故效率最佳。为了使用一个共享内存段,一般需要以下几个步骤:

  1. 调用shmget()创建一个新共享内存段或获取已有共享内存段的标识符。调用完成返回一个后续需要用到的共享内存标识符
  2. 调用shmat()用将shmid (shmget生成的)标识的共享内存段附加到调用进程的虚拟地址空间中
  3. 到此,程序可以将其与其他内存一样对待。shmat()系统调用会返回该段在虚拟地址空间的始址
  4. 调用shmdt()分离共享内存段,这样进厂将无法继续引用这段空间(可选)
  5. 调用shmctl()删除共享内存段,只有一个进程执行,当所有附加该段到虚拟地址空间的进程都分离之后才能将其销毁
进程1 内核 共享内存区 进程2 shmget(key, size, IPC_CREAT|0666) 返回 shmid shmat(shmid, NULL, 0) 返回内存指针 shm_ptr1 shmget(key, size, 0) 返回相同 shmid shmat(shmid, NULL, 0) 返回内存指针 shm_ptr2 创建物理内存段 写入数据 (e.g., sprintf(shm_ptr1, "Hello")) 读取数据 (e.g., printf("%s", shm_ptr2)) shmdt(shm_ptr1) shmdt(shm_ptr2) shmctl(shmid, IPC_RMID, NULL) 标记为待删除 当所有进程断开后 内核释放物理内存 进程1 内核 共享内存区 进程2

在这里插入图片描述


02. 共享内存接口

2.1 共享内存数据结构

struct shmid_ds {struct ipc_perm shm_perm;  // 权限信息size_t          shm_segsz; // 段大小(字节)time_t          shm_atime; // 最后附加时间time_t          shm_dtime; // 最后分离时间time_t          shm_ctime; // 最后修改时间pid_t           shm_cpid;  // 创建者PIDpid_t           shm_lpid;  // 最后操作PIDshmatt_t        shm_nattch;// 当前附加计数
};

2.2 生成键值key

       #include <sys/types.h>#include <sys/ipc.h>
//将`路径名`和`项目ID`转换成与之对应的`IPC`键值key_t ftok(const char *pathname, int proj_id);

**返回值:**蔡成功返回一个key_t值,失败返回-1

在这里插入图片描述


2.3 创建/获取共享内存段

       #include <sys/ipc.h>#include <sys/shm.h>int shmget(key_t key, size_t size, int shmflg);

参数:

  • key:

    • IPC_PRIVATE(创建新内存)
    • 使用ftok() 生成的键值(用于进程间共享)
  • size:共享内存大小>0

  • shmflg

    • IPC_CREAT(不存在时创建)
    • IPC_EXCL(若存在则失败)
    • 权限位(如 0666

返回值:成功返回共享内存标识符 shmid,失败返回 -1

//part of code   
key_t key = ftok("../test", 'a');int shmid = shmget(key, 1024, IPC_CREAT | 0666);  // 不存在时创建   cout <<"shmid is:"<< shmid << endl;

在这里插入图片描述


2.4 连接共享内存

       #include <sys/types.h>#include <sys/shm.h>
//将共享内存段映射到进程的地址空间void *shmat(int shmid, const void *shmaddr, int shmflg);

参数:

  • shmid: 由shmget()返回的共享内存标识符
  • shmaddr:请求的附加地址:
    • NULL(由系统自动选择地址)
    • 显示指定(不推荐)
  • shmflg
    • SHM_RDONLY(只读)
    • SHM_REMAP(Linux下覆盖现有映射)
    • 0默认行为

返回值:成功返回共享内存的起始地址指针,失败返回 (void *) -1

//同一进程,没啥用  ,只是验证  
char*ptr=(char*)shmat(shmid, NULL, 0);sprintf(ptr, "Hello!");cout << ptr << endl;

2.5 控制共享内存

       #include <sys/ipc.h>#include <sys/shm.h>int shmctl(int shmid, int cmd, struct shmid_ds *buf);

参数:

  • shmid:共享内存标识符

  • cmd(可选):

    • IPC_RMID(标记删除,最后一个进程分离后生效)
    • IPC_STAT(获取状态到 buf
    • IPC_SET(通过 buf 设置参数)
  • buf:用于存储或修改共享内存属性的结构体(cmd后两个选项)

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


2.6 分离共享内存

       #include <sys/types.h>#include <sys/shm.h>
//解除进程与共享内存段的映射关系int shmdt(const void *shmaddr);//0 success - failed

参数:

  • shmaddr:由shmat()返回的共享内存始址

返回值:成功返回0,失败返回 -1`

注: 分离一个共享内存段删除它是不同的。删除是通过shmctl()选项IPC_ RMID操作来完成的。


03. C/S简单通信

3.2 server.cpp

#include "comm.h"
int main(){key_t key = ftok(PATH_NAME, PROJ_ID);if(key < 0){perror("ftok");return 1;}int shmId = shmget(key, SIZE,IPC_CREAT | IPC_EXCL | 0666);//创建全新的shm,如果和系统已经存在ID冲突,就出错返回if(shmId < 0){perror("shmget");return 2;}printf("key->%u, shmId->%d\n", key, shmId);//sleep(1);char* mem = (char*)shmat(shmId, NULL, 0);//建立关联printf("attaches shm success\n");//sleep(15);//开始逻辑实现的部分while(1){sleep(1);printf("%s\n", mem);}  shmdt(mem);//去关联printf("detaches shm success\n");shmctl(shmId, IPC_RMID, NULL);//sleep(5);printf("key->0x%u, shmId->%d   shm delete success\n", key, shmId);//sleep(10);return 0;
}

3.2 client.c

#include "comm.h"
int main(){key_t key = ftok(PATH_NAME, PROJ_ID);if(key < 0){perror("ftok");return 1;}printf("%u\n",key);//client这里只需要获取即可int shmid = shmget(key, SIZE, IPC_CREAT);if(shmid < 0){perror("shmget");return 1;}char* mem = (char*)shmat(shmid, NULL, 0);//建立关联//sleep(5);printf("client process attaches success!\n");//这里是进行通信的部分char c = 'A';while(c <= 'Z'){mem[c - 'A'] = c;c++;mem[c - 'A'] = 0;sleep(2);}shmdt(mem);//移除关联//slee(5);  printf("client process detaches success!\n");return 0;
}
http://www.xdnf.cn/news/750475.html

相关文章:

  • Python应用for循环遍历寻b
  • homework 2025.03.31 chinese(class 3)
  • DeepSeek R1-0528:深度思考能力的重大跃升与技术突破全解析
  • 一文读懂Automotive SPICE
  • Day41 Python打卡训练营
  • 【HW系列】—Windows日志与Linux日志分析
  • 远程线程注入
  • 【PhysUnits】15.5 引入P1后的标准化表示(standardization.rs)
  • Cesium快速入门到精通系列教程一
  • AtCoder AT_abc408_d [ABC408D] Flip to Gather
  • C++ 变量声明(Declaration)和定义(Definition)的区别
  • 【系统配置与部署类】linux系统下的desktop图标文件配置
  • 如何配置国内docker镜像源?
  • leetcode3128. 直角三角形-medium
  • [VMM]现代 CPU 中用于加速多级页表查找的Page‐Table Entry原理
  • 人工智能在智能健康监测中的创新应用与未来趋势
  • could not select device driver ““ with capabilities: [[gpu]]
  • 红外遥控(外部中断)
  • 关于win10系统中环境变量path变成一行显示的问题
  • Vue ①-实例 || 指令
  • Baklib企业CMS全流程管控与智能协作
  • CppCon 2014 学习:Optimization Tips
  • Fine Pruned Tiled Light Lists(精细删减的分块光照列表)
  • Python60日基础学习打卡Day39
  • 痉挛性斜颈带来的困扰
  • PCIE之Lane Reserval通道out of oder调换顺序
  • 2025年- H63-Lc171--33.搜索旋转排序数组(2次二分查找,需二刷)--Java版
  • 基于热力学熵增原理的EM-GAM
  • Less基础语法
  • Python打卡训练营学习记录Day41