直接来看共享内存的操作函数:
功能 创建共享内存,也用来获取共享内存的id值
头文件 #include<sys/shm.h>
函数原型
int shmget(key_t key, size_t size, int shmflg);
key 创建共享内存的关键字
相同的key值可获取相同的共享内存ID
size 共享内存的大小
shmflg 共享内存的使用权限
注意:
使用了SHM_RDONLY(4096),则以只读方式使用此共享内存段
否则,以读写的方式进行连接
返回值 >0 共享内存ID
-1 失败
功能 将共享内存附加到进程空间中(也可认为为获取共享内存地址)
头文件 #include<sys/shm.h>
函数原型
void * shmat(int shmid, void *shmaddr, int shmflg);
shmid 共享内存ID
shmaddr 指定共享内存在当前进程中的地址
通常将shmaddr指定为0,表示将此段连接到由内核选择的第一个可用地址上
shmflg 共享内存的使用权限
返回值 >0 共享内存的首地址
-1 失败
功能 脱解共享内存段(针对用shmat进行添加的进程)
头文件 #include<sys/shm.h>
函数原型
int shmdt(void *addr);
addr 调用shmat时返回的共享内存首地址
返回值 0 成功
-1 失败
功能 对共享内存执行各种操作
头文件 #include<sys/shm.h>
函数原型
int shmctl(int shmid,int cmd,struct shmid_ds *buf);
shmid 共享内存ID
cmd 对共享内存要执行的操作
IPC_STAT 获取共享内存段的shmid_ds结构,将其存放在由buf
指向的结构中。
IPC_SET 按buf指向结构中的值设定此段相关结构中的
shm_perm.uid、 shm_perm.gid、
shm_perm.mode
IPC_RMID 从系统中删除共享内存段
SHM_LOCK 将共享内存锁定在内存中
SHM_UNLOCK 解锁共享内存段
buf 设置共享内存时需要的数据或接收共享内存属性的内存地址
返回值 0 成功
-1 失败
接着来看一个实例,服务器获取当前的系统时间利用共享内存存储,客户端对共享内存的内容进行接受并在终端显示。
//客户端
#include <stdio.h>
#include <stdlib.h>
#include <sys/shm.h>
#include <time.h>
#include <unistd.h>
#define SHM_KEY 99
int main()
{
int seg_id;
char *mem_ptr;
int i=30;
//获取共享内存的id值
if((seg_id=shmget(SHM_KEY,100,0777))==-1)
{
perror("shmget");
exit(1);
}
while(i>0){
//根据共享内存的id获取共享内存的首地址
mem_ptr=(char *)shmat(seg_id,0,SHM_RDONLY);
if(mem_ptr==NULL)
{
perror("shmat");
exit(1);
}
printf("The time from server is:%s\n",mem_ptr);
sleep(1);
i--;
}
//脱解共享内存段(与共享内存断开连接)
shmdt(mem_ptr);
}
//服务器
#include <stdio.h>
#include <stdlib.h>
#include <sys/shm.h>
#include <time.h>
#include <string.h>
#include <unistd.h>
#define SHM_KEY 99
int main()
{
int seg_id;
char *mem_ptr;
time_t now;
int i=30;
//创建共享内存并获取其id
seg_id=shmget(SHM_KEY,100,IPC_CREAT|0777);
if(seg_id==-1)
{
perror("shmget");
exit(1);
}
//获取共享内存的地址值
mem_ptr=(char *)shmat(seg_id,0,0);
if(mem_ptr==NULL)
{
perror("shmat");
exit(1);
}
while(i>0)
{
time(&now);
puts(ctime(&now));
//将内容写入共享内存的地址(首地址不变,内容会进行覆盖)
strcpy(mem_ptr,ctime(&now));
sleep(1);
i--;
}
//创建者对系统中该共享内存进行删除
shmctl(seg_id,IPC_RMID,NULL);
return 0;
}
关于共享内存信号量的加入,请参考: Linux共享内存的使用(二)