【操作系统】进程间的通信——共享内存

简介: 【操作系统】进程间的通信——共享内存

进程间的通信-共享内存

共享内存机制

共享内存机制是允许两个或多个进程(不相关或有亲缘关系)访问同一逻辑内存的机制。它是共享和传递数据的一种非常有效的方式。不同进程之间共享的内存通常安排为同一段物理内存。

两种常用的共享内存方式

  • System V版本的共享内存 shmm

    • 多进程直接共享内存
  • 文件映射mmap

    • 如果一个文件需要频繁进行读写,那么将它映射到内存中。
    • 将特殊文件进行匿名内存映射,为有关联的进程提供共享内存空间。
    • 为无关联的进程提供共享内存空间,将一个普通文件映射到内存中。

System V版本的共享内存 shmm

原理:

利用共享内存完成进程间通信,两个进程都可以通过虚拟地址空间到用户页表,然后通过用户级页表映射到物理内存的相同一块内存区域。

image-20220824183601419

共享内存的使用

ftok

  • 作用:创建一个唯一的key,来标识一块共享内存。(用不用都行)
  • 函数原型:key_t ftok(const char *pathname,int proj_id)
  • 参数:使用下面两个创建一个唯一的key

    • pathname:
    • proj_id:
  • 返回值:返回一个key。

shmget

  • 作用:创建一个共享内存块,返回这个共享内存块的标识符shmid。
  • 函数原型:int shmget(key_t key,size_t size,int shmflg)
  • 参数说明:

    • key:由ftok函数生成。
    • size:申请的共享内存的大小,为4k的整数倍。因为x86系列的cpu在linux系统下面基本上,内存也都是以4kb为单位。
    • shmflg:权限标识。详见-shmget(2) — Linux manual page

      • IPC_CREAT:不存在则创建。
      • IPC_EXCL:与IPC_CREAT一起使用,若以存在则创建失败。
  • 返回值:返回创建的这个共享内存块的标识符shmid
  • 相关参考


shmat

  • 作用:挂接共享内存,将共享内存挂接到当前进程的地址空间,即允许本进程访问一块共享内存。
  • 函数原型:void shmat(int shmid,const void shmaddr, int shmflg)
  • 参数:

    • shmid:挂接的共享内存ID,由shmget函数获得。
    • shmaddr: 一般为0,表示连接到由内核选择的第一个可用地址上。
    • shmflg:标记,一般为0。
  • 返回值:

    • 指向内存地址的指针。
  • 相关参考:


shmdt

  • 作用:取消共享内存映射
  • 函数原型:int shmdt(const void *shmaddr);
  • 参数:

    • shmaddr:由shmat返回的指向内存的指针。
  • 返回值:

    • 成功,则返回0
    • 失败,则返回-1,并设置errno。
  • 相关参考:


shmctl

  • 作用: 用于控制共享内存。
  • 函数原型:int shmctl(int shmid, int cmd, struct shmid_ds *buf);

    • shmid:共享内存标识码,由shmget获取的。
    • cmd:将要采取的动作(可取值:IPC_STAT、IPC_SET、IPC_RMID...)。
    • buf:指向着一个保存着共享内存的模式状态和访问权限的数据结构。

示例

shmwrite.c

#include <unistd.h>
#include <stdlib.h>
#include <stdio.h>
#include <string.h>
#include <sys/shm.h>

struct Conn_stat{
    int count;
    char ip[64];
};

int main(void){

    void *shm = NULL;
    int shmid = 0;//存储共享内存的标识
    struct Conn_stat stat = {0,"127.0.0.1"};

    //创建共享内存
    shmid = shmget((key_t)1234,sizeof(struct Conn_stat),0666 | IPC_CREAT);
    if(shmid == -1){
        fprintf(stderr, "shmget failed\n");
        exit(1);
    }
    //将共享内存挂接到当前进程的地址空间
    shm = shmat(shmid,(void*)0,0);
    if(shm == (void*)-1){
        fprintf(stderr, "shmat failed\n");
        exit(2);
    }
    printf("Memory attached at %p\n", shm);
    //设置共享内存
    struct Conn_stat *p = (struct Conn_stat*)shm;
    memcpy(p,&stat,sizeof(struct Conn_stat));
    //修改共享内存中的数据
    int i = 0;
    while((i++) < 50){
        p->count++;
        sleep(1);
    }
    //把共享内存从当前进程中分离
    if(shmdt(shm) == -1){
        fprintf(stderr, "shmdt failed\n");
        exit(3);
    }
    return 0;
}

shmread.c

#include <unistd.h>
#include <stdlib.h>
#include <stdio.h>
#include <sys/shm.h>
#include <string.h>
#include <errno.h>

struct Conn_stat{
    int count;
    char ip[64];
    
};

int main(void){

    void *shm = NULL;//分配的共享内存的原始首地址
    struct Conn_stat *stat = NULL;
    int shmid;//共享内存标识符
    //创建共享内存
    shmid = shmget((key_t)1234, sizeof(struct Conn_stat), 0666|IPC_CREAT);
    if(shmid == -1){
        fprintf(stderr, "shmget failed\n");
        exit(0);
    }
    //将共享内存挂接到当前当前进程的地址空间
    shm = shmat(shmid, 0, 0);
    if(shm == (void*)-1){
        fprintf(stderr, "shmat failed\n");
        exit(1);
    }
    printf("\nMemory attached at %p\n", shm);
    //设置共享内存
    stat = (struct Conn_stat*)shm;
    int i = 0;
    while((i++) < 10){
        printf("ip = %s ,count: %d\t\t\n", stat->ip, stat->count);    
        sleep(1);
    }
    //把共享内存从当前进程中分离
    if(shmdt(shm) == -1){
        fprintf(stderr, "shmdt failed\n");
        exit(2);
    }
    //删除共享内存
    if(shmctl(shmid, IPC_RMID, 0) == -1){
        fprintf(stderr, "shmctl(IPC_RMID) failed, reason: %s\n",strerror(errno));
        exit(3);
    }
    return 0;   
}

image-20220825090353811


存储映射共享I/O mmap

原理:

将一个普通文件或者其它对象映射进内存。

  • 使用普通文件提供的内存映射。
  • 使用特殊文件提供匿名内存映射。

image-20220825101107429


使用

mmap

  • 作用:mmap是一种内存映射文件的方法,即将一个文件或者其它对象映射到进程的地址空间,实现文件磁盘地址和进程虚拟地址空间中一段虚拟地址的一一对映关系。
  • 函数原型: void mmap(void addr, size_t length, int prot, int flags,int fd, off_t offset);
  • 参数:

    • addr:指向欲映射的内存起始地址,通常设为NULL,代表让系统自动选定地址,映射成功后返回该地址。
    • length:将文件中多大的部分映射到内存。
    • prot:映射区域的保护方式。可以为以下几种方式的组合:

      • PROT_EXEC:执行
      • PROT_READ:读取
      • PROT_WRITE:写入
      • PROT_NONE:不能存取
    • flags:影响映射区域的各种特性。

      • MAP_SHARED:映射区域数据与文件对应,允许其他进程共享。
      • MAP_PRIVATE:映射区域生成文件的copy,修改不同步文件。
      • MAP_ANONYMOUS:建立匿名映射。此时会忽略参数fd,不涉及文件,而且映射区域无法和其他进程共享。
      • MAP_DENYWRITE:允许对映射区域的写入操作,其他对文件直接写入的操作将会被拒绝。
      • MAP_LOCKED:将映射区域锁定住,这表示该区域不会被置swap。
    • fd: 要映射到内存中的文件描述符。

      • 如果使用匿名内存映射时,即flags中设置了MAP_ANONYMOUS,fd设为-1。
      • 有些系统不支持匿名内存映射,则可以使用fopen打开/dev/zero文件,然后对该文件进行映射,可以同样达到匿名内存映射的效果。
    • offset:文件映射的偏移量,通常设置为0。

      • 代表从文件最前方开始对应,offset必须是分页大小的整数倍。
  • 返回值:

    • 成功:返回指向映射区域的指针。
    • 失败:MAP_FAILED,其值为(void *)-1,并设置errno。
  • 相关参考:


munmap

  • 作用:解除映射。
  • 函数原型: int munmap(void *addr, size_t length); 
  • 参数:

    • addr: 由mmap返回的地址。
    • length:映射区的大小。
  • 返回值:

    • 成功:返回0。
    • 失败:返回-1,并设置errno。
  • 相关参考:


示例

map_write.c

#include<stdio.h>
#include<stdlib.h>
#include<unistd.h>
#include<fcntl.h>
#include<sys/types.h>
#include<sys/stat.h>
#include<sys/mman.h>
#include<string.h>

struct Conn_stat{
    int count;
    char ip[64];
};

//这个进程创建映射区进行写
int main(int argc,char* argv[]){
    if(argc != 2){
        printf("Usage: %s  file.\n",argv[0]);
        exit(1);
    }
    struct Conn_stat stat =  {0,"127.0.0.1"};
    //打开文件
    int fd = open(argv[1],O_RDWR|O_CREAT|O_TRUNC,0644);
    if(fd<0){
        perror("open");
        exit(2);
    }
    //将文件截断为第二个参数这么大
    ftruncate(fd,sizeof(struct Conn_stat)); 
    //创建一个结构体大小的共享映射区。共享映射区我们可以当做数组区看待。
    struct Conn_stat *p = (struct Conn_stat*)mmap(NULL,sizeof(struct Conn_stat),PROT_READ|PROT_WRITE,MAP_SHARED,fd,0);
    if(p == MAP_FAILED){
        perror("mmap");
        exit(3);
    }
    close(fd);//关闭不用的文件描述符
    memcpy(p,&stat,sizeof(struct Conn_stat));
    while(p->count < 30){
        p->count++;
        sleep(1);
    }
    //解除映射
    int ret = munmap(p,sizeof(struct Conn_stat));
    if(ret < 0){
        perror("mmumap");
        exit(4);
    }

    return 0;
}

map_read.c

#include<stdio.h>
#include<stdlib.h>
#include<unistd.h>
#include<fcntl.h>
#include<sys/types.h>
#include<sys/stat.h>
#include<sys/mman.h>

struct Conn_stat{    
    int count;
    char ip[64];
};

int main(int argc,char *argv[]){
    if(argc != 2){
        printf("Usage: %s  file.\n",argv[0]);
        exit(1);
    }
    //打开文件-只读,注意与写中的参数不同
    int fd = open(argv[1],O_RDONLY,0644);
    if(fd < 0){
        perror("open");
        exit(2);
    }
    
    struct Conn_stat stat;
    //注意与写中的参数不同
    struct Conn_stat *p = (struct Conn_stat*)mmap(NULL,sizeof(struct Conn_stat),PROT_READ,MAP_SHARED,fd,0);
    if(p == MAP_FAILED){
        perror("mmap");
        exit(3);
    }
    close(fd);
    int i = 0;
    while((i++) < 10){
        printf("ip = %s ,count: %d\t\t\n",p->ip,p->count);    
        sleep(1);
    }
    //接触映射
    int ret = munmap(p,sizeof(stat));
    if(ret < 0){
        perror("mmumap");
        exit(4);
    }
    return 0;
}

image-20220825112012906

相关文章
|
6天前
|
算法 调度 UED
探索操作系统核心:进程管理与调度
【9月更文挑战第28天】在数字世界的心脏跳动着无数进程,它们像是细胞一样构成了操作系统的生命体。本文将深入探讨操作系统中进程管理与调度的奥秘,揭示如何通过精心设计的数据结构和算法来维护系统的稳定性和效率。我们将从进程的基本概念出发,逐步解析进程状态转换、进程同步机制,以及进程调度策略,旨在为读者呈现一幅清晰、生动的操作系统内部工作机制图景。
|
2天前
|
算法 调度 UED
探索操作系统的心脏:进程调度算法
【9月更文挑战第32天】在数字世界的每一次心跳中,都隐藏着一个不为人知的英雄——进程调度算法。它默默地在后台运作,确保我们的命令得到快速响应,应用程序平稳运行。本文将带你走进操作系统的核心,一探进程调度的奥秘,并通过代码示例揭示其背后的智慧。准备好跟随我一起深入这趟技术之旅了吗?让我们开始吧!
|
4天前
|
算法 Linux 调度
深入理解操作系统的进程调度
【9月更文挑战第30天】本文将带你进入操作系统的核心—进程调度。我们将探讨其工作原理,分析几种常见的调度算法,并通过实际代码示例来揭示这些理论是如何在真实系统中实现的。无论你是初学者还是有经验的开发者,这篇文章都能帮助你更好地理解操作系统的这一关键组成部分。
|
4天前
|
消息中间件 算法 调度
探索操作系统核心:进程管理与调度策略
【9月更文挑战第30天】在数字化时代的心脏,操作系统扮演着至关重要的角色。本文将深入探讨操作系统的基石之一——进程管理,以及如何通过调度策略优化系统性能。我们将从进程的基本概念出发,逐步解析进程状态、进程控制和进程间通信等关键要素。同时,我们会探讨几种常见的进程调度算法,并分析它们的优缺点。最后,文章将展示一个简单的代码示例,以加深对理论部分的理解和应用。
|
5天前
|
算法 调度 UED
探索操作系统的心脏:进程管理与调度
【9月更文挑战第29天】在数字世界的海洋中,操作系统是支撑软件与硬件和谐共舞的桥梁。本文将深入探讨操作系统的核心功能—进程管理及其调度机制,揭示它们是如何影响计算机性能和用户体验的。通过浅显易懂的语言和生动的比喻,我们将一起遨游在进程的生命周期、调度算法以及优先级等概念之间,旨在为读者呈现一个清晰的操作系统内部运作图景。
17 6
|
4天前
|
算法 调度 开发者
深入理解操作系统之进程管理与调度
【9月更文挑战第30天】本文旨在通过浅显易懂的语言和具体代码示例,带领读者探索操作系统中进程管理的奥秘。我们将从进程的生命周期出发,逐步解析进程调度的核心概念,并通过实例展示如何实现简单的进程调度算法。无论你是初学者还是有一定基础的开发者,都能在这篇文章中找到有价值的信息,帮助你更好地理解和掌握进程管理与调度的知识。
15 4
|
7天前
|
算法 调度
操作系统的心脏:深入解析进程调度算法
本文旨在深入探讨现代操作系统中的核心功能之一——进程调度。进程调度算法是操作系统用于分配CPU时间片给各个进程的机制,以确保系统资源的高效利用和公平分配。本文将详细介绍几种主要的进程调度算法,包括先来先服务(FCFS)、短作业优先(SJF)、时间片轮转(RR)以及优先级调度(PS)。我们将分析每种算法的基本原理、优缺点及其适用场景。同时,本文还将讨论多级反馈队列(MFQ)调度算法,并探讨这些算法在实际应用中的表现及未来发展趋势。通过深入解析这些内容,希望能够为读者提供对操作系统进程调度机制的全面理解。
|
6天前
|
资源调度 算法 调度
深入浅出操作系统之进程与线程管理
【9月更文挑战第29天】在数字世界的庞大舞台上,操作系统扮演着不可或缺的角色,它如同一位精通多门艺术的导演,精心指挥着每一个进程和线程的演出。本文将通过浅显的语言,带你走进操作系统的内心世界,探索进程和线程的管理奥秘,让你对这位幕后英雄有更深的了解。
|
5天前
|
算法 调度 UED
深入理解操作系统:进程管理与调度策略
【9月更文挑战第29天】在数字世界的心脏,操作系统悄无声息地跳动着,它的健康直接关系到整个计算生态系统的活力。本文将带领读者穿梭于操作系统的微观世界,探索进程管理的奥秘和调度策略的智慧。我们将从进程的基本概念出发,逐步深入到进程的生命周期管理,最后探讨不同的进程调度算法及其对系统性能的影响。通过深入浅出的方式,让读者能够更好地理解并掌握操作系统中进程管理的核心知识。
|
6天前
|
算法 Linux 调度
深入理解操作系统中的进程调度
【9月更文挑战第28天】在操作系统的复杂世界中,进程调度是维持系统高效运作的关键。本文将深入浅出地探讨进程调度的核心概念及其对系统性能的影响。从进程调度的定义和目标出发,逐步解析不同类型的调度算法,并通过实际代码示例,揭示这些算法如何在真实系统中实施。无论你是初学者还是有经验的开发者,这篇文章都将为你提供宝贵的见解和知识。

热门文章

最新文章

下一篇
无影云桌面