共享内存是最快的一种IPC形式;因为消息队列,管道等都需要用到系统接口,而共享内存则是直接改变逻辑地址到物理地址的映射关系,即当一个进程A看到这片内存时,对应的只要改变进程B的映射关系,此时进程B就可以立马看到这份公共资源;
1.共享内存的原理:
(1)在物理内存上开辟一块内存空间;
(2)将这块内存通过页表映射到进程的虚拟地址空间;
(3)进程可以直接通过进程的虚拟地址访问到这块物理内存,进行操作(若多个进程映射同一块物理内存,就可以实现相互通信—直接通过虚拟地址来改变内存中的数据,其他进程也会随之改变,相较于其他进程间通信方式,少了两Ubuntu内核态与用户态之间的数据拷贝过程)因此速度快;
(4)解除映射关系;
(5)删除共享内存;
原理图如下所示:
2.共享内存函数:
(1)创建内存:
int shmget(key_t key, size_t size, int shmflg);
参数:
key:共享内存在操作系统中的标识符;
size:共享内存的大小;
shmflg:包含两部分;
IPC_CREAT:共享内存存在则打开,否则创建;
IPC_EXCL:与IPC_CREAT同时使用,若共享内存存在则报错返回;
返回值:成功:正整数; 失败:-1;
(2)映射虚拟地址空间:
void *shmat(int shmid, const void *shmaddr, int shmflg);
参数:
shmid:创建共享内存返回的操作句柄;
shmaddr:共享内存在虚拟地址空间中的首地址----通常置NULL;
shmflg:SHM_RDONLY---映射之后,共享内存只读;通常置0-可读可写;
返回值:映射首地址 (void*) -1
(3)共享内存的操作:
memcpy/strcpy;
(4)解除映射关系:
int shmdt(const void *shmaddr);
shmaddr:shmat建立映射时返回的映射首地址;
(5)删除共享内存:
int shmctl(int shmid, int cmd, struct shmid_ds *buf);
shmid:共享内存操作句柄;
cmd:即将进行的操作;
IPC_RMID 删除共享内存;
buf:获取/设置共享内存信息;
共享内存的删除流程:共享内存在删除的时候,首先会判断当前的映射链接数是否为0;若为0则直接删除;否则表示现在还有其他进程正在使用,则共享内存不能被立即删除,但是会拒绝后续进程的映射链接,等待映射链接数为0时删除;
ipcs可查看当前系统上的进程间通信方式;
3.关于共享内存的代码实现:
在两个端口下进行操作:
shm_read.c文件下:
#include <stdio.h>
#include <unistd.h>
#include <errno.h>
#include <stdlib.h>
#include <sys/shm.h>
#include <string.h>
#define IPC_KEY 0x12345678
int main(){
int shmid=shmget(IPC_KEY,32,IPC_CREAT | 0664);
if(shmid<0){
perror("shmget error");
return -1;
}
void* shm_start=shmat(shmid,NULL,0);
if(shm_start<0){
perror("shmat error");
return -1;
}
while(1){
printf("%s\n",shm_start);
sleep(1);
}
shmdt(shm_start);
shmctl(shmid,IPC_RMID,NULL);
return 0;
}
shm_write文件下:
#include <stdio.h>
#include <unistd.h>
#include <string.h>
#include <errno.h>
#include <stdlib.h>
#include <sys/shm.h>
#define IPC_KEY 0x12345678
int main(){
int shmid=shmget(IPC_KEY,32,IPC_CREAT | 0664);
if(shmid<0){
perror("shmget error");
return -1;
}
void* shm_start=shmat(shmid,NULL,0);
if(shm_start<0){
perror("shmat error");
return -1;
}
int i=0;
while(1){
sprintf(shm_start,"%s-%d","晚上没饭吃~~~",i++);
sleep(1);
}
shmdt(shm_start);
shmctl(shmid,IPC_RMID,NULL);
程序执行结果:
第一个端口进行写操作:
第二个端口进行读操作: