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

Linux——共享内存

目录

一、共享内存概念

 二、共享内存的一些函数

2.1 shmget 创建共享内存

2.2 shmat 访问共享内存 

2.3 shmdt 解除共享内存的映射

2.4 shnctl 删除共享内存段

三、共享内存 

3.1 创建测试进程

3.2 使用循环测试

​编辑

3.3 共享内存写入程序

3.4 带有信号量的共享内存 


一、共享内存概念

共享内存是一种进程间通信的方式,允许多个进程访问和操作同一块内存区域。这样的内存区域被所有共享它的进程所拥有,进程可以将数据写入共享内存区域,也可以从中读取数据。共享内存在提高进程通信效率和降低开销方面具有优势,但也需要进行同步和互斥操作以避免数据竞争和冲突。

共享内存通常适用于需要高效地在进程间传递大量数据的场景,比如多个进程需要共享大型数据结构、图形图像处理或多媒体应用等。在使用共享内存时,需要注意管理内存的权限、同步访问和处理异常情况等问题,以确保数据的一致性和安全性。

 二、共享内存的一些函数

2.1 shmget 创建共享内存

它的函数原型如下:

int shmget(key_t key, size_t size, int shmflg);

参数含义:

1. key: 用于标识共享内存段的键值,不同的进程使用相同的 key 值可以获取到同一个共享内存
2. size: 创建共享内存时,指定要申请的共享内存空间大小
3. shmflg: 用于指定创建共享内存段的访问权限和其他标志,比如权限位和内存段的创建方式等。常见的标志包括IPC_CREAT(如果不存在则创建共享内存段)和 IPC_EXCL(如果存在则返回错误)。
shmget() 成功则返回共享内存的 ID, 失败返回-1
#include<stdio.h>
#include<stdlib.h>
#include<unistd.h>
#include<string.h>
#include<sys/shm.h>int main()
{int shmid = shmget((key_t)2345,128,IPC_CREAT|0600);if(shmid == -1){exit(1);}}

2.2 shmat 访问共享内存 

函数shmat()用于将共享内存段附加到调用进程的地址空间,并返回一个指向共享内存段起始地址的指针。

它的函数原型如下:

void* shmat(int shmid, const void *shmaddr, int shmflg);

参数含义:

  1. shmid: 表示要附加的共享内存段的标识符,通常是由shmget()函数返回的共享内存标识符。

  2. shmaddr: 指定共享内存段连接到调用进程地址空间的地址。一般设置为 NULL,由系统自动选择映射的虚拟地址空间

  3. shmflg: 用以指定附加共享内存段的附加方式。常见的标志包括SHM_RDONLY(只读方式附加共享内存段)和 SHM_RND(指示共享内存段将在系统限定的地址范围内附加)。 一般给 0, 可以给 SHM_RDONLY 为只读模式,其他的为读写

返回值:

       函数的返回值是一个指针,指向共享内存段的起始地址。  shmat()成功则返回共享内存的首地址,失败返回 NULL
#include<stdio.h>
#include<stdlib.h>
#include<unistd.h>
#include<string.h>
#include<sys/shm.h>int main()
{int shmid = shmget((key_t)2345,128,IPC_CREAT|0600);if(shmid == -1){exit(1);}char* s=(char*)shmat(shmid,NULL,0);if(s==(char*)-1){exit(1);}strcpy(s,"hello");shmdt(s);}

2.3 shmdt 解除共享内存的映射

当进程不再需要访问共享内存时,需要使用 shmdt 函数将共享内存段从进程地址空间中解除映射。其函数原型如下:

int shmdt(const void *shmaddr);
参数含义:
  1. shmaddr: 表示要分离的共享内存段的起始地址。通常是由shmat()函数返回的指针,指向共享内存段的起始位置。

返回值:

        当成功分离共享内存段时,函数返回0。如果出现错误,函数会返回-1,并且可以通过检查errno变量来获取错误信息。

2.4 shnctl 删除共享内存段

函数shmctl()用于控制共享内存段的属性

int shmctl(int shmid, int cmd, struct shmid_ds *buf);

参数含义:

  1. shmid: 共享内存段的标识符,通常由shmget()函数返回。

  2. cmd: 控制命令,指定对共享内存段执行的操作类型。常见的命令有:

    • IPC_STAT: 获取共享内存段的状态信息,并将其存储在buf指向的shmid_ds结构体中。
    • IPC_SET: 设置共享内存段的权限模式,需要提供buf指向的shmid_ds结构体。
    • IPC_RMID: 从系统中删除共享内存段,同时释放其占用的资源。
  3. buf: 指向一个shmid_ds结构体的指针,用于存储或传递共享内存段的状态信息或权限设置。

返回值:

        函数成功执行时返回0,否则返回-1,并通过设置errno变量来指示错误类型。

三、共享内存 

3.1 创建测试进程

#include<stdio.h>
#include<stdlib.h>
#include<unistd.h>
#include<string.h>
#include<sys/shm.h>int main()
{int shmid = shmget((key_t)2345,128,IPC_CREAT|0600);if(shmid == -1){exit(1);}char* s=(char*)shmat(shmid,NULL,0);if(s==(char*)-1){exit(1);}printf("s=%s\n",s);// 打印共享内存的地址shmdt(s);}

运行结果:

3.2 使用循环测试

#include<stdio.h>
#include<stdlib.h>
#include<unistd.h>
#include<string.h>
#include<sys/shm.h>int main()
{int shmid = shmget((key_t)2345,128,IPC_CREAT|0600);if(shmid == -1){exit(1);}char* s=(char*)shmat(shmid,NULL,0);if(s==(char*)-1){exit(1);}while(1){if(strncmp(s,"end",3) == 0){break;}printf("s=%s\n",s);sleep(1);}shmdt(s);}

3.3 共享内存写入程序

#include<stdio.h>
#include<stdlib.h>
#include<unistd.h>
#include<string.h>
#include<sys/shm.h>int main()
{int shmid = shmget((key_t)2345,128,IPC_CREAT|0600);if(shmid == -1){exit(1);}char* s=(char*)shmat(shmid,NULL,0);if(s==(char*)-1){exit(1);}while(1){printf("input\n");char buff[128]={0};fgets(buff,128,stdin);strcpy(s,buff);if( strncmp(buff,"end",3) == 0){break;}}shmdt(s);
}

3.4 带有信号量的共享内存 

首先要创建两个信号量,一个设为0,另一个设为1。

头文件代码sem.h

#include<stdio.h>
#include<stdlib.h>
#include<unistd.h>
#include<string.h>
#include<sys/sem.h>enum INDEX{SEM1=0,SEM2};union semun
{int val;
};void sem_init();//创建并初始化信号量
void sem_p(int index);
void sem_v(int index);
void sem_destroy();

sem.c代码

#include"sem.h"static int semid=-1;void sem_init()//创建并初始化信号量
{semid=semget((key_t)1234,2,IPC_CREAT|IPC_EXCL|0600);if(semid==-1){semid=semget((key_t)1234,2,0600);if(semid==-1){printf("semget err\n");}}else{int arr[2]={1,0};for(int i=0;i<2;i++){union semun a;a.val=arr[i];if(semctl(semid,i,SETVAL,a)==-1){printf("semctl init err\n");}}}
}void sem_p(int index)
{struct sembuf buf;buf.sem_num=index;buf.sem_op=-1;//pbuf.sem_flg=SEM_UNDO;if(semop(semid,&buf,1) == -1){printf("op p err\n");}
}void sem_v(int index)
{struct sembuf buf;buf.sem_num=index;buf.sem_op=1;//vbuf.sem_flg=SEM_UNDO;if(semop(semid,&buf,1) == -1){printf("op v err\n");}
}void sem_destroy()
{if(semctl(semid,0,IPC_RMID) == -1){printf("semctl del\n");}
}

使用信号量的main.c代码

#include<stdio.h>
#include<stdlib.h>
#include<unistd.h>
#include<string.h>
#include<sys/shm.h>
#include"sem.h"int main()
{int shmid = shmget((key_t)2345,128,IPC_CREAT|0600);if(shmid == -1){exit(1);}char* s=(char*)shmat(shmid,NULL,0);if(s==(char*)-1){exit(1);}sem_init();//创建 初始化while(1){printf("input\n");char buff[128]={0};fgets(buff,128,stdin);sem_p(SEM1);//p  s1strcpy(s,buff);sem_v(SEM2);//v  s2if( strncmp(buff,"end",3) == 0){break;}}shmdt(s);
}

测试代码test.c

#include<stdio.h>
#include<stdlib.h>
#include<unistd.h>
#include<string.h>
#include<sys/shm.h>
#include"sem.h"int main()
{int shmid = shmget((key_t)2345,128,IPC_CREAT|0600);if(shmid == -1){exit(1);}char* s=(char*)shmat(shmid,NULL,0);if(s==(char*)-1){exit(1);}sem_init();while(1){sem_p(SEM2);if(strncmp(s,"end",3) == 0){break;}printf("s=%s\n",s);sem_v(SEM1);}shmdt(s);sem_destroy();exit(0);
}

运行结果:

相关文章:

  • 2025年第十六届蓝桥杯省赛JavaB组真题回顾
  • 威锋VL822-Q7T10GHUB芯片适用于扩展坞显示器
  • Java异常报错:java.nio.channels.UnresolvedAddressException
  • 在 IntelliJ IDEA 中开发 Java Web 项目时,遇到包内明明存在某个类但类名仍然爆红(显示红色错误提示)
  • 广汽滴滴Robotaxi首次亮相,中国自动驾驶加速领跑新赛道
  • 明天见!奇兵到家将携“偏远服务第一网”登陆成都建博会
  • Linux常见指令介绍上(入门级)
  • ubuntu 2204 安装 vcs 2023
  • LangChain4j (3):集成 DeepSeek 大模型的Java程序
  • 质量问题频发,如何提升源头把控
  • classpath “com.android.tools.build:gradle:8.1.0“ 和 Gradle插件版本8.5 有什么关系
  • KMP算法核心笔记:前后缀本质与nextval实现
  • GitHub配置密钥
  • 环境变量概念以及获取环境变量(linux下解析)
  • C#Winform程序将子窗体嵌入容器方法
  • 15、nRF52xx蓝牙学习(串口输入与回环)
  • code review时线程池的使用
  • Oracle数据库数据编程SQL<00. 课外关注:rownum、rowid、level、row_number 对比详解与实战>
  • Mac关闭sip方法
  • MySQL-运维篇
  • 富力地产:广州富力空港假日酒店第一次拍卖流拍,起拍价约2.77亿元
  • 全国人大常委会调研组在宁波调研,张庆伟带队钟山易炼红参加
  • 智飞生物一季度营收下滑79%,连续三个季度亏损,称业绩波动与行业整体趋势一致
  • 对话地铁读书人|豪宅房产经纪人:读书使我免于抑郁
  • 在因关税战爆火的敦煌网上,美国人爱买什么中国商品
  • 根据学习教育安排,上海市委中心组专题学习总书记力戒形式主义官僚主义重要论述