共享内存
原理
- 共享内存为多个进程之间的共享和传递数据提供了一种有效的方式。
- 共享内存是先在物理内存上申请一块空间,多个进程可以将其映射到自己的虚拟地址空间中。所有进程都可以访问共享内存中的地址。如果某个进程向共享内存中写入数据,所做的改动将立刻被可以访问同一段共享内存的任何其他进程看到。
- 由于他并未提供同步机制,所以我们通常需要用其他的机制来同步对共享内存的访问。
接口介绍
头文件
#include<sys/ipc.h>
#include<sys/shm.h>
#include<sys/types.h>
接口
int shmget(key_t key,size_t size,int shmflg);
shmget()用于创建或者获取共享内存
shmget()成功返回共享内存的ID,失败返回-1
key:不同的进程使用相同的key值可以获取到同一个共享内存
size:创建共享内存时,指定要申请的共享内存空间大小
shmflg:IPC_CREAT IPC_EXCL
void* shmat(int shmid,const void *shmaddr,int shmflg);
shmat()将申请的共享内存的物理地址映射到当前进程的虚拟地址空间上
shmat()成功返回共享内存的首地址,失败返回NULL
shmaddr:一般给NULL,由系统自动选择映射的虚拟地址空间
shmflg:一般给0,可以给SHM_RDONLY为只读模式,其他为读写
int shmdt(const void* shmaddr);
shmdt()断开当前进程的shmaddr指向的共享内存映射
shmdt()成功返回0,失败返回-1
int shmctl(int shmid,int cmd,struct shmid_fs *buf);
shmctl()控制共享内存
shmctl()成功返回0,失败返回-1
cmd:IPC_RMID
示例代码
共享内存的创建和映射
题目描述:进程a从键盘循环获取数据并拷贝到共享内存中,进程b从共享内存中获取并打印数据。要求进程a输入一次,进程输出一次,进程a不输入,进程b也不输出。
思路:
情况一:设定一个信号量,可以实现,但是在数据写入过快的情况下,存在读取不及时的情况,无法实现写入一个读取一个。
情况二:设定两个信号量s1和s2,初始设定:s1=1,s2=0。
申请信号量s1资源,进行写入,s1-1==0;
vs2释放信号量s2,s2+1=1,此时对s2可以进行P操作,进行数据打印;
数据打印结束,对s1进行资源释放,以便于写操作的进行。
代码实现:
#include<stdio.h>
#include<stdlib.h>
#include<string.h>
#include<unistd.h>
#include<sys/shm.h>
int main()
{
int shmid=shmget((key_t)1234,128,IPC_CREAT|0600);
if(shmid==-1)
{
exit(1);
}
char* s=(char*)shmat(shmid,NULL,0);
}