Linux进程间通信之共享内存

时间:2023-03-09 19:09:04
Linux进程间通信之共享内存

一,共享内存
  内核管理一片物理内存,允许不同的进程同时映射,多个进程可以映射同一块内存,被多个进程同时映射的物理内存,即共享内存。
  映射物理内存叫挂接,用完以后解除映射叫脱接

1,共享内存的特点:

  优点:是最快的IPC。
  缺点:要编程者自己实现对共享内存互斥访问。如何实现?

2,编程模型:具体函数的用法可以用man手册查看(强力推荐)

进程A: writeshm.c
     1) 获得key, ftok()
     2) 使用key来创建一个共享内存 shmget()
     3) 映射共享内存(得到虚拟地址), shmat()
     4) 使用共享内存, 往共享内存中写入数据
     5) 解除映射 shmdt()
     6) 如果共享内存不再使用,可以使用shmctl()销毁共享内存

进程B: readshm.c

  1) 获得key, ftok()

  2) 使用key来获得一个共享内存 shmget()

  3) 映射共享内存(得到虚拟地址), shmat()

  4) 使用共享内存, 读取共享内存中的数据

  5) 解除映射 shmdt()

3,实例

进程A:

// writeshm.c

#include<stdio.h>
#include<unistd.h>
#include<stdlib.h>
#include<sys/types.h>
#include<sys/ipc.h>
#include<sys/shm.h> int main()
{
// 生成一个key
key_t key = ftok("./", ); // 创建共享内存,返回一个id
int shmid = shmget(key, , IPC_CREAT||IPC_EXCL);
if(- == shmid)
{
perror("shmget failed");
exit();
} // 映射共享内存,得到虚拟地址
void *p = shmat(shmid, , );
if((void*)- == p)
{
perror("shmat failed");
exit();
} // 写共享内存
int *pp = p;
*pp = 0x12345678;
*(pp + ) = 0xffffffff; // 解除映射
if(- == shmdt(p))
{
perror("shmdt failed");
exit();
}
printf("解除映射成功,点击回车销毁共享内存\n");
getchar(); // 销毁共享内存
if(- == shmctl(shmid, IPC_RMID, NULL))
{
perror("shmctl failed");
exit();
} return ;
}

进程B:

// readshm.c

#include<stdio.h>
#include<unistd.h>
#include<stdlib.h>
#include<sys/types.h>
#include<sys/ipc.h>
#include<sys/shm.h> int main()
{
// 生成一个key
key_t key = ftok("./", ); // 获取共享内存,返回一个id
int shmid = shmget(key, , );
if(- == shmid)
{
perror("shmget failed");
exit();
} // 映射共享内存,得到虚拟地址
void *p = shmat(shmid, , );
if((void*)- == p)
{
perror("shmat failed");
exit();
} // 读共享内存
int x = *(int *)p;
int y = *((int *)p + );
printf("从共享内存中都取了:0x%x 和 0x%x \n", x, y); // 解除映射
if(- == shmdt(p))
{
perror("shmdt failed");
exit();
} return ;
}

运行结果:

writeshma:

Linux进程间通信之共享内存

readshma:

Linux进程间通信之共享内存