前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >Linux 进程间如何共享内存?

Linux 进程间如何共享内存?

作者头像
用户6543014
发布2019-10-25 18:01:33
8.4K0
发布2019-10-25 18:01:33
举报
文章被收录于专栏:CU技术社区

导读

共享内存是在内存中单独开辟的一段内存空间,这段内存空间有自己特有的数据结构,包括访问权限、大小和最近访问的时间等。

共享内存 IPC 原理

共享内存进程间通信机制主要用于实现进程间大量的数据传输,下图所示为进程间使用共享内存实现大量数据传输的示意图:

共享内存是在内存中单独开辟的一段内存空间,这段内存空间有自己特有的数据结构,包括访问权限、大小和最近访问的时间等。该数据结构定义如下:

代码语言:javascript
复制
from /usr/include/linux/shm.h
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 最近attach时间 */
__kernel_time_t shm_dtime; /* last detach time 最近detach时间 */
__kernel_time_t shm_ctime; /* last change time 最近change时间 */
__kernel_ipc_pid_t shm_cpid; /* pid of creator 创建者pid */
__kernel_ipc_pid_t shm_lpid; /* pid of last operator 最近操作pid */
unsigned short shm_nattch; /* no. of current attaches */
unsigned short shm_unused; /* compatibility */
void *shm_unused2; /* ditto - used by DIPC */
void *shm_unused3; /* unused */|
};

两个进程在使用此共享内存空间之前,需要在进程地址空间与共享内存空间之间建立联系,即将共享内存空间挂载到进程中。

系统对共享内存做了以下限制:

代码语言:javascript
复制
#define SHMMAX 0x2000000 /* max shared seg size (bytes) 最大共享段大小 */

#define SHMMIN 1 /* min shared seg size (bytes) 最小共享段大小 */

#define SHMMNI 4096 /* max num of segs system wide */

#define SHMALL (SHMMAX/getpagesize()*(SHMMNI/16))|

define SHMSEG SHMMNI /* max shared segs per process */

Linux 共享内存管理

1.创建共享内存

代码语言:javascript
复制
#include <sys/ipc.h> #include <sys/shm.h>

/* * 第一个参数为 key 值,一般由 ftok() 函数产生 * 第二个参数为欲创建的共享内存段大小(单位为字节) * 第三个参数用来标识共享内存段的创建标识 */

代码语言:javascript
复制
int shmget(key_t key, size_t size, int shmflg);

2.共享内存控制

代码语言:javascript
复制
#include <sys/ipc.h> #include <sys/shm.h>

/* * 第一个参数为要操作的共享内存标识符 * 第二个参数为要执行的操作 * 第三个参数为 shmid_ds 结构的临时共享内存变量信息 */

代码语言:javascript
复制
int shmctl(int shmid, int cmd, struct shmid_ds *buf);

3.映射共享内存对象

系统调用 shmat() 函数实现将一个共享内存段映射到调用进程的数据段中,并返回内存空间首地址,其函数声明如下:

代码语言:javascript
复制
#include <sys/types.h>

#include <sys/shm.h>

/* * 第一个参数为要操作的共享内存标识符 * 第二个参数用来指定共享内存的映射地址,非0则为此参数,为0的话由系统分配 * 第三个参数用来指定共享内存段的访问权限和映射条件 */

代码语言:javascript
复制
void *shmat(int shmid, const void *shmaddr, int shmflg);

4.分离共享内存对象

在使用完毕共享内存空间后,需要使用 shmdt() 函数调用将其与当前进程分离。函数声明如下:

代码语言:javascript
复制
#include <sys/types.h>

#include <sys/shm.h>

/* * 参数为分配的共享内存首地址 */

代码语言:javascript
复制
int shmdt(const void *shmaddr);

共享内存在父子进程间遵循的约定

1.使用 fork() 函数创建一个子进程后,该进程继承父亲进程挂载的共享内存。

2.如果调用 exec() 执行一个新的程序,则所有挂载的共享内存将被自动卸载。

3.如果在某个进程中调用了 exit() 函数,所有挂载的共享内存将与当前进程脱离关系。

程序实例

申请一段共享内存,父进程在首地址处存入一整数,子进程读出。

代码语言:javascript
复制
#include

#include <sys/ipc.h>

#include <sys/shm.h>

#include <sys/types.h>

#include

#include

#define SHM_SIZE 1024

int main()

{

int shm_id, pid;

int *ptr = NULL;

/* 申请共享内存 */

代码语言:javascript
复制
shm_id = shmget((key_t)1004, SHM_SIZE, IPC_CREAT | 0600);

/* 映射共享内存到进程地址空间 */

代码语言:javascript
复制
ptr = (int*)shmat(shm_id, 0, 0);

printf("Attach addr is %p \n", ptr);

*ptr = 1004;

printf("The Value of Parent is : %d \n", *ptr);

if((pid=fork()) == -1){

perror("fork Err");

exit(0);

}

else if(!pid){
printf("The Value of Child is : %d \n", *ptr);
exit(0);
}else{
sleep(1);

/* 解除映射 */

代码语言:javascript
复制
shmdt(ptr);

/* 删除共享内存 */

代码语言:javascript
复制
shmctl(shm_id, IPC_RMID, 0);
}
return 0;
}
输出结果:
本文参与 腾讯云自媒体同步曝光计划,分享自微信公众号。
原始发表:2019-06-12,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 SACC开源架构 微信公众号,前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档