【Linux】进程间通信----共享内存

1.共享内存

共享存储允许两个或多个进程共享一给定的存储区。因为数据不需要在客户机和服务器之间复制,所以这是最快的一种 IPC。

使用共享存储的唯一窍门是多个进程之间对一给定存区的同步存取。若服务器将数据放入共享存储区,则在服务器做完这一操作之前,客户机不当去取这些数据。

 

2.共享内存原理图

在Linux中,每个进程都有属于自己的进程控制块(PCB)和地址空间(Addr Space),并且都有一个与之对应的页表,负责将进程的虚拟地址与物理地址进行映射,通过内存管理单元(MMU)进行管理。两个不同的虚拟地址通过页表映射到物理空间的同一区域,它们所指向的这块区域即共享内存。

  • 原理
    在这里插入图片描述
  • 在物理内存当中开辟了一段空间
  • 这段物理内存的空间,可以被不同的进程附加到自己的共享区当中
  • 附加的进程,通过操作共享区,来交换数据
     

3.共享内存的接口

  • 创建共享内存
int shmget(key_t key, size_t size, int shmflg);

参数

  • key:共享内存标识符,这个标识符相当于共享内存的身份证,程序员第一次创建的时候,可以随意给值,只要和当前操作系统当中的其他的共享内存标识符不重复
  • size:共享内存的大小,单位:字节
  • shmflg:① IPC_CREAT:如果共享内存不存在,则创建共享内存
    IPC_EXCL:需要搭配IPC_CREAT使用,这样的宏在搭配使用的时候,还是采用按位或的方式(核心就是位图)
    IPC_CREAT | IPC_EXCL
      如果想要获取的共享内存,已经存在,则报错。
      如果想要获取的共享内存,是刚刚新创建出来的共享内存,则返回操作句柄

返回值

  • 返回的是共享内存的操作句柄

总结

  • 使用shmget这个函数的时候一定要自己刚刚创建出来的共享内存

 

  • 将共享内存附加到进程
void* shmat(int shmid, const void *shmaddr, int shmflg);

参数

  • shmid:共享内存的操作句柄
  • shmaddr:附加到共享内存的什么虚拟地址,允许传递NULL值让操作系统帮我们选择附加到共享内存区当中的那个地址,这个地址通过该函数的返回值返回给我们
  • shmflg:① SHM_RDONLY:规定当前进程只能对共享内存进行读操作
    0可读可写
  • 返回值:返回共享内存附加到共享区的地址
  • 结论:进程在读取共享内存的时候,是访问,并不是拿走!
     
  • 将共享内存和进程分离
int shmdt(const void *shmaddr)

参数

  • shmaddr:shmat的返回值
     
  • 操作共享内存
int shmctl(int shmid, int cmd, struct shmid_ds *buf)

参数

  • shmid:共享内存操作句柄
  • cmd:告诉shmctl函数需要做什么操作
    IPC_STAT:获取当前共享内存的属性信息,放在buf中,buf是出参
    IPC_SET:设置共享内存的属性信息,是用buf来进行设置的,buf是入参
    IPC_RMID:删除共享内存,buf可以直接传递NULL
    buf:共享内存的结构体

 

  • 共享内存命令
  • ipcs:查看共享内存
  • ipcrm -m [shmid]:删除共享内存
  • 共享内存的生命周期是跟随着操作系统内核的
     

4.共享内存的标识符和操作句柄的区别

  • 标识符:是用来标识共享内存的,相当于共享内存的身份证,意味着不同的进程可以通过标识符找到这个共享内存
  • 操作句柄:进程可以通过操作句柄来对共享内存进行操作(附加,分离,删除)
     

5.共享内存释放

  • 当使用shmctl或者使用ipcrm,删除共享内存之后,共享内存就实际被释放
  • 当共享内存被释放掉之后,共享内存的标识符会被设置为0x00000000,表示其他进程不能通过之前的标识符找到该共享内存,并且共享内存的状态会被设置为dest(destroy)
  • 共享内存被释放掉之后,但是还是有进程在附加着共享内存,当前描述共享内存的结构体并没有被释放,直到当前共享内存的附加进程数量为0的时候才会释放掉

 

6.shmid_ds结构

内核为每个共享存储段设置了一个shmid_ds结构。

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 */
};
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值