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

共享内存是最快的一种IPC形式;因为消息队列,管道等都需要用到系统接口,而共享内存则是直接改变逻辑地址到物理地址的映射关系,即当一个进程A看到这片内存时,对应的只要改变进程B的映射关系,此时进程B就可以立马看到这份公共资源;

1.共享内存的原理:

(1)在物理内存上开辟一块内存空间;
(2)将这块内存通过页表映射到进程的虚拟地址空间;
(3)进程可以直接通过进程的虚拟地址访问到这块物理内存,进行操作(若多个进程映射同一块物理内存,就可以实现相互通信—直接通过虚拟地址来改变内存中的数据,其他进程也会随之改变,相较于其他进程间通信方式,少了两Ubuntu内核态与用户态之间的数据拷贝过程)因此速度快;
(4)解除映射关系;
(5)删除共享内存;

原理图如下所示:
在这里插入图片描述

2.共享内存函数:

(1)创建内存:
int shmget(key_t key, size_t size, int shmflg);

参数:

key:共享内存在操作系统中的标识符;
size:共享内存的大小;
shmflg:包含两部分;
    IPC_CREAT:共享内存存在则打开,否则创建;
	IPC_EXCL:与IPC_CREAT同时使用,若共享内存存在则报错返回;

返回值:成功:正整数; 失败:-1;

(2)映射虚拟地址空间:
void *shmat(int shmid, const void *shmaddr, int shmflg);

参数:

shmid:创建共享内存返回的操作句柄;
shmaddr:共享内存在虚拟地址空间中的首地址----通常置NULL;
shmflg:SHM_RDONLY---映射之后,共享内存只读;通常置0-可读可写;

返回值:映射首地址 (void*) -1

(3)共享内存的操作:
memcpy/strcpy;

(4)解除映射关系:
int shmdt(const void *shmaddr);

shmaddr:shmat建立映射时返回的映射首地址;

(5)删除共享内存:
int shmctl(int shmid, int cmd, struct shmid_ds *buf);

shmid:共享内存操作句柄;
cmd:即将进行的操作;
    IPC_RMID  删除共享内存;
	
buf:获取/设置共享内存信息;

共享内存的删除流程:共享内存在删除的时候,首先会判断当前的映射链接数是否为0;若为0则直接删除;否则表示现在还有其他进程正在使用,则共享内存不能被立即删除,但是会拒绝后续进程的映射链接,等待映射链接数为0时删除;

ipcs可查看当前系统上的进程间通信方式;
在这里插入图片描述

3.关于共享内存的代码实现:

在两个端口下进行操作:

shm_read.c文件下:
  #include <stdio.h>
  #include <unistd.h>
  #include <errno.h>                                                                                                                  
  #include <stdlib.h>
  #include <sys/shm.h>
  #include <string.h> 
  #define IPC_KEY 0x12345678
  int main(){
    int shmid=shmget(IPC_KEY,32,IPC_CREAT | 0664);
    if(shmid<0){
      perror("shmget error");
      return -1;
    } 
    void* shm_start=shmat(shmid,NULL,0);
    if(shm_start<0){
      perror("shmat error");
      return -1;
    }
  
    while(1){
      printf("%s\n",shm_start);
      sleep(1);
    }
  
    shmdt(shm_start);
    shmctl(shmid,IPC_RMID,NULL);
    return 0;
 }
 

shm_write文件下:
  #include <stdio.h>
  #include <unistd.h>
  #include <string.h>
  #include <errno.h>
  #include <stdlib.h>
  #include <sys/shm.h>
  #define IPC_KEY 0x12345678
  int main(){
    int shmid=shmget(IPC_KEY,32,IPC_CREAT | 0664);
    if(shmid<0){
      perror("shmget error");
      return -1;
    }
  
    void* shm_start=shmat(shmid,NULL,0);
    if(shm_start<0){
      perror("shmat error");
      return -1;
    } 
   
    int i=0;                                                                                                                          
    while(1){
      sprintf(shm_start,"%s-%d","晚上没饭吃~~~",i++);
      sleep(1);
    }
    shmdt(shm_start);
    shmctl(shmid,IPC_RMID,NULL);

程序执行结果:

第一个端口进行写操作:
在这里插入图片描述

第二个端口进行读操作:
在这里插入图片描述

  • 3
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值