共享内存区是最快的IPC形式。一旦这样的内存映射到共享他的进程的地址空间,这些进程间数据传递不再涉及到内核,换句话说就是进程不再通过执行进入内核的系统调用来传递彼此的数据。
共享内存示意图
共享内存数据结构
struct shmid_ds {
struct ipc_perm shm_perm; /* operation perms */
int shm_segsz; /* size of segment (bytes) */
__kernel_time_t shm_atime; /* last attach time */
__kernel_time_t shm_dtime; /* last detach time */
__kernel_time_t shm_ctime; /* last change time */
__kernel_ipc_pid_t shm_cpid; /* pid of creator */
__kernel_ipc_pid_t shm_lpid; /* pid of last operator */
unsigned short shm_nattch; /* no. of current attaches */
unsigned short shm_unused; /* compatibility */
void *shm_unused2; /* ditto - used by DIPC */
void *shm_unused3; /* unused */
};
相关函数
shmget函数:创建共享内存
int shmget(key_t key,size_t size,int shmflg);
//参数:key为共享内存段名字
//size为共享内存大小
//shmflg有9个权限标志构成,他们的用法和创建文件时使用的mode模式标志是一样的
//返回值:成功返回一个非负整数,即该共享内存段的标识码;失败返回-1
shmat函数 :将共享内存段链接到进程地址空间
void *shmat(int shmid,const void *shmaddr,int shmflg);
//参数:shmid共享内存标识符
//shmaddr:指定连接的地址
//shmflg:他的两个可能取值是SHM_RND和SHM_RDONLY
//返回值:成功返回一个指针,指向共享内存第一个节;失败返回-1
1: shmaddr为NULL,核心自动选择一个地址
2:shmaddr不为NULL且shmflg无SHM_RND标记,则以shmaddr为连接地址
3:shmaddr不为NULL且shmflg设置了SHM_RND标记,则连接的地址会自动向下调整为SHMLBA的整数倍,公式:shmaddr - (shmaddr&SHMLBA)
4: shmflg=SHM_RDONLY,表示连接操作用来只读共享内存
shmdt函数:将共享内存段与当前进程脱离
int shmdt(const void *shmaddr);
//参数:shmaddr是由shmat所返回的指针
//返回值:成功返回0,失败返回-1
//需要注意一点:将共享内存段与当前进程脱离不代表删除共享内存段
shmctl函数:控制共享内存
int shmctl(int shmid,int cmd,struct shmid_ds *buf);
//参数:shmid由shmget返回的共享内存标识码
//cmd:将要采取的动作(有3个可取值)
//buf:指向一个保存着共享内存的模式状态和访问权限的数据结构
//返回值:成功返回0,失败返回-1
cmd取值
1、IPC_STAT:把shmid_ds结构中的数据设置为共享内存的当前关联值
2、IPC_SET:在金城有足够权限的前提下,把共享内存的当前关联值设置为shmid_ds数据结构中给出的值
3、IPC_RMID:删除共享内存段
说明几点:
1:共享内存没有提供任何保护机制
2:共享内存是临界资源
3:共享内存生命周期随内核(消息队列也是)
4:共享内存没有进行同步与互斥