一、【共享内存的概述】
共享内存允许两个或者多个进程共享指定的存储区域
共享内存的特点:
1、共享内存是进程间共享数据的一种最快的方法。一个进程向共享的内存区域写入了数据,共享这个内存区域的所有进程就可以立刻看到其中的内容。
2、使用共享内存要注意的是多个进程之间对一个给定存储区访问的互斥。若一个进程正在向共享内存区写数据,则在它做完这一步操作前,别的进程不应当去读、写这些数据
二、【共享内存的API】
1、创建共享内存区shmget()函数
头文件:
#include <sys/ipc.h>
#include <sys/shm.h>
函数原型:
int shmget(key_t key, size_t size, int shmflg);
这里需要获取key,我们要先使用ftok函数去获取唯一的key值。
key_t key = ftok("/",xxxx);
功能:
创建或打开一块共享内存区
参数:
key: IPC键值 需要使用ftok函数去获取
size:该共享存储段的长度(字节)
shmflg:标识函数的行为及共享内存的权限:
IPC_CREAT:如果不存在就创建
IPC_EXCL:如果已经存在则返回失败
位或权限位:共享内存位或权限位后可以设置共享内存的访问权限,格式和open函数的mode_t一样,但可执行权限未使用
返回值:
成功:返回共享内存标识符
失败:返回-1
2、共享区映射shmat函数
头文件:
#include <sys/types.h>
#include <sys/shm.h>
函数原型:
void *shmat(int shmid, const void *shmaddr, int shmflg);
功能:
将一个共享内存映射到调用进程的数据段中。
参数:
shmid :共享内存标识符
shmaddr:共享内存映射地址:如果是NULL就系统自动指定。
shmflg:.共享内存段的访问权限和映射条件:
0:共享内存具有可读可写权限。
SHM_RDONLY:只读。
SHM_RND: (shmaddr非空时才有效)
没有指定.SHM.RND则此段连接到shmaddr.所指定的地址上( shmaddn必须页对齐)。
指定工.SHM_RND: 则此段连接到shmaddr- shmaddn%SHMLBA 所表示的地址上
返回值:
成功:返回共享内存段映射地址
失败:.返回-1
3、解除共享区映射区shmdt函数
头文件:
#include <sys/types.h>
#include <sys/shm.h>
函数原型:
int shmdt(const void *shmaddr);
功能:
将共享内存和当当前进程分离(仅仅是断开联系并不删除共享内存)。
参数:
shmaddr:共享内存映射地址。
返回值:
成功返回0
失败返回-1。
4、控制共享内存shmctl函数
头文件:
#include <sys/ipc.h>↓
#include <sys/shm.h>↓
函数原型:
int shmctl(int shmid, int cmd,struct shmid_ds*buf) ;↓
功能:
共享内存空间的控制。
参数:
shmid:.共享内存标识符。
cmd:函数功熊的控制。
buf:shmid_ds数据类型的地址,用来存放或修改共享内存的属性
cmd:函数功能的控制
IPC_RMID:删除。
IPC_SET:设置shmid_ds参数
IPC_STAT:保存shmid_ds参数
SHM_LOCK:锁定共享内存段(超极用户)
SHM_UNLOCK:解锁共享内存段。
返回值:
成功返回0,失败返回-1
三、案例:创建共享内存,一个进程负责读,一个进程负责写
#include <stdio.h>
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/shm.h>
#include<string.h>
#include <unistd.h>
#include <sys/ipc.h>
#include<wait.h>
int main(int argc, char const *argv[])
{
pid_t pid = fork();
if(pid==-1)
{
perror("fork");
return -1;
}
//子进程负责写
if(pid==0)
{
//获取key值
key_t key = ftok("/",2098);
//创建共享内存
int shm_id = shmget(key,64,IPC_CREAT|0666);
//共享内存映射
char *buf = (char*)shmat(shm_id,NULL,0);
strcpy(buf,"hello yz1!");
shmdt(buf);
}
//父进程负责收
if(pid>0)
{
//获取key值
key_t key = ftok("/",2098);
//创建共享内存
int shm_id = shmget(key,64,IPC_CREAT|0666);
//共享内存映射
char *buf = (char*)shmat(shm_id,NULL,0);
printf("父进程:打印buf内容:%s\n",buf);
shmdt(buf);
wait(NULL);
}
return 0;
}