什么是共享内存?
通俗来讲,一个男的和一个女的坐在一张桌子上,桌子上有一张纸,男的在这张纸上写了一段文字给女的,女的刚好可以看得见。换成进程来讲,就是A进程往一块公共内存里写了一段数据,然后B进程刚好可以从这一块公共内存里看到A进程写的数据。同样B进程也可以往公共存储区里写数据,A进程也可以从公共存储区里读取数据。这个过程大致可以描述共享内存的原理。共享内存,就是一个或多个进程共享一个给定的存储区。
共享内存的API
// 创建或获取一个共享内存:成功返回共享内存ID,失败返回-1
int shmget(key_t key, size_t size, int flag);
// 连接共享内存到当前进程的地址空间:成功返回指向共享内存的指针,失败返回-1
void *shmat(int shm_id, const void *addr, int flag);
// 断开与共享内存的连接:成功返回0,失败返回-1
int shmdt(void *addr);
// 控制共享内存的相关信息:成功返回0,失败返回-1
int shmctl(int shm_id, int cmd, struct shmid_ds *buf);
如何实现共享内存
流程
- 创建或打开共享内存
- 将共享内存映射到进程中
- 写数据/读数据
- 释放共享内存
- 干掉共享内存
demo
下面的demo实现了两个进程用共享内存的方式进行通信
demoshm_read.c
#include <stdio.h>
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/shm.h>
#include <string.h>
#include <stdlib.h>
#include <unistd.h>
/*
共享内存的流程:
1. 创建或打开共享内存
2. 将共享内存映射到进程中
3. 写数据/读数据
4. 释放共享内存
5. 干掉共享内存
*/
int main()
{
key_t key;
int shmid;
char *shmaddr;
key = ftok(".",1);
printf("key is :%x\n",key);
shmid = shmget(key,1024*4,0);
if(shmid == -1){
printf("shm failed!\n");
exit(-1);
}
shmaddr = shmat(shmid,0,0);
printf("shmat over!\n");
printf("It says : %s\n",shmaddr);
shmdt(shmaddr);
printf("quit\n");
return 0;
}
demoshm_send.c
#include <stdio.h>
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/msg.h>
#include <string.h>
struct msgbuf {
long mtype; /* message type, must be > 0 */
char mtext[30]; /* message data */
};
int main()
{
key_t key;
struct msgbuf sendbuf={888,"Welcome to the linux"};
//专门用来获取IPC的ID的
key=ftok(".",'Z');
printf("key is %x\n",key);
int msgid = msgget(key,IPC_CREAT|0700);
if(msgid == -1){
printf("creat msg failed!\n");
}
//发送信息
msgsnd(msgid,&sendbuf,sizeof(sendbuf.mtext),0);
struct msgbuf readbuf;
msgrcv(msgid,&readbuf,sizeof(readbuf.mtext),988,0);
printf("Return from read: %s\n",readbuf.mtext);
return 0;
}