作用
多个进程间通信
特点
- 效率最高;
- 在操作共享内存时,记得加互斥;
- 可以传递自定义消息;
- 消息不是一次性的;
- 消息传递实时(由于是修改地址内容来传递消息,不及时读取,之前的消息会被覆盖)。
命令
查看共享内存 ipcs -m
删除共享内存 ipcrm -m 共享内存号
查看共享内存限制值 ipcs -ml
函数
1、创建并打开共享内存区(物理内存上)
#include <sys/ipc.h>
#include <sys/shm.h>
int shmget(key_t key, size_t size,int shmflg);
功能:创建或打开一块共享内存区
参数:
key:IPC键值
size:该共享存储段的长度(字节)
shmflg:标识函数的行为及共享内存的权限。
参数:
shmflg:
IPC_CREAT:如果不存在就创建
IPC_EXCL:如果已经存在则返回失败
位或权限位:共享内存位或权限位后可以设置共享内存的访问权限,格式和open函数的mode_t一样,但可执行权限未使用。
返回值:
成功:返回共享内存标识符。
失败:返回-1。
2、建立连接(映射关系)
#include <sys/types.h>
#include <sys/shm.h>void *shmat(int shmid, const void *shmaddr, int shmflg);
功能: 将一个共享内存段映射到调用进程的数据段中。
参数:
shmid:共享内存标识符。
shmaddr:共享内存映射地址(若为NULL则由系统自动指定),推荐使用NULL。
shmflg:共享内存段的访问权限和映射条件
0:共享内存具有可读可写权限。
SHM_RDONLY:只读。
SHM_RND:(shmaddr非空时才有效)
没有指定SHM_RND则此段连接到shmaddr所指定的地址上(shmaddr必需页对齐)。
指定了SHM_RND则此段连接到shmaddr- shmaddr%SHMLBA 所表示的地址上。
返回值:
成功:返回共享内存段映射地址
失败:返回 -1
注意: shmat函数使用的时候第二个和第三个参数一般设为NULL和0,即系统自动指定共享内存地址,并且 共享内 存可读可写。
3、解除链接关系
#include <sys/types.h>
#include <sys/shm.h>int shmdt(const void *shmaddr);
功能:
将共享内存和当前进程分离(仅仅是断开联系并不删除共享内存)。
参数:
shmaddr:共享内存映射地址。
返回值:
成功返回 0,失败返回 -1。
4、共享内存控制
#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。
5、实例
//shm.c
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/shm.h>
#include <stdio.h>
#include <string.h>
#include <unistd.h>
#include <fcntl.h>
int main()
{
//创建共享内存
key_t key = ftok("./", 2023);
int shmid = shmget(key, 1024, 0666 | IPC_CREAT);
if(shmid<0)
return 0;
//建立连接
void *str = shmat(shmid, NULL, 0);
//写入数据
strcpy(str, "删库跑路!!!!!!");
//断开连接
shmdt(str);
//删除共享内存
//shmctl(shmid, IPC_RMID, NULL);
return 0;
}
//shmrcv.c
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/shm.h>
#include <stdio.h>
#include <string.h>
#include <unistd.h>
#include <fcntl.h>
int main()
{
//创建共享内存
key_t key = ftok("./", 2023);
int shmid = shmget(key, 1024, 0666 | IPC_CREAT);
if(shmid<0)
return 0;
//建立连接
void *str = shmat(shmid, NULL, 0);
//读数据
printf("%s\n",(char*)str);
//断开连接
shmdt(str);
//删除共享内存(选用)
shmctl(shmid, IPC_RMID, NULL);
return 0;
}