当前位置: 首页 > 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;
}

相关文章:

  • Python 从入门到精通视频下载
  • 各种数据库,行式、列式、文档型、KV、时序、向量、图究竟怎么选?
  • 点云识别模型汇总整理
  • 【Doris基础】Doris中的Replica详解:Replica原理、架构
  • 华为OD机试真题——找出两个整数数组中同时出现的整数(2025A卷:100分)Java/python/JavaScript/C++/C语言/GO六种最佳实现
  • 黄金价格查询接口如何用C#进行调用?
  • Nacos实战——动态 IP 黑名单过滤
  • AI书签管理工具开发全记录(七):页面编写与接口对接
  • 手写HashMap
  • AE已禁用刷新请释放Caps Lock
  • 现代网络安全攻防技术与发展现状
  • 头歌java课程实验(学习-Java字符串之正则表达式之元字符之判断字符串是否符合规则)
  • 使用Python实现Windows系统垃圾清理
  • Webug4.0靶场通关笔记16- 第16关MySQL配置文件下载
  • 项目日记 -Qt音乐播放器 -搜索模块
  • Linux研学-用户解析
  • 【Java笔记】Spring IoC DI
  • ApiHug 1.3.9 支持 Spring 3.5.0 + Plugin 0.7.4 内置小插件升级!儿童节快乐!!!
  • 新闻数据加载(鸿蒙App开发实战)
  • flowable候选人及候选人组(Candidate Users 、Candidate Groups)的应用包含拾取、归还、交接
  • 代做课程设计的网站/上海谷歌seo
  • 上海平台网站建设企业/seo的定义
  • 做幼儿网站的目标/seo的形式有哪些
  • 建设项目竣工验收公示网站/网络推广推广外包服务
  • 专业做旅游网站的公司/防疫管控优化措施
  • 湛江网站建设开发/色盲眼镜