Linux 共享内存mmap,进程通信

简介: Linux 共享内存mmap,进程通信

前言

进程间通信是操作系统中重要的概念之一,使得不同的进程可以相互交换数据和进行协作。其中,共享内存是一种高效的进程间通信机制,而内存映射(mmap)是实现共享内存的一种常见方法。


一、存储映射 I/O

存储映射 I/O 是 一个磁盘文件 与 存储空间中的一个缓冲区相映射。于是, 当从缓冲区中取数据,就相当于读文件中的相应字节。于此类似,将数据存入缓冲区,则相应的字节就自动写入文件。这样, 就可在 不适用 read 和 write 函数的情况下,使用 地址(指针)完成 i/o 操作。

使用这种方法,首先应通知内核,将一个指定文件映射到存储区域中。这个映射工作可以通过 mmap 函数来实现。使用 mmap 系统调用,进程可以直接操作共享内存的指针,而不需要复杂的数据结构和同步机制。

理解 共享内存

共享内存是一种特殊的内存区域,它可以被多个进程访问和操作。这意味着不同的进程可以直接读取或写入该共享内存区域中的数据。相比于其他进程间通信机制,共享内存具有较低的开销和高效的数据传输速度。


二、mmap, munmap

mmap 用于创建共享内存映射。munmap 用来 释放内存。

 #include <sys/mman.h>
 void *mmap(void *addr, size_t length, int prot, int flags,
           int fd, off_t offset);
 int munmap(void *addr, size_t length);
  1. void * mmap ( void * addr, size_t length , int prot , int flags , int fd , off_t offset ) ;

参数:

  • addr : 指定映射区的首地址。通常传 NULL / 0,表示让系统自动分配。
  • length :共享映射区的大小。
  • prot : 共享映射区的读写属性。
  • flags : 标注共享内存的共享属性。
  • fd :用于创建共享映射区的哪个文件的,文件描述符。
  • offset :

返回值:

  • 成功 : 映射区的首地址。
  • 失败 : 返回 M AP_FAILED。
  1. int munmap ( void * addr , size_t length ) ;

三、父子进程间 mmap 通信

void sys_error(const char *str)
{
  perror(str);    
  exit(1);                  // 正常退出程序
}
int var = 10;
int main(void)
{
  int fd;
  char *p;
  pid_t pid;
  fd = open("1.txt", O_RDWR);
  if(fd < 0)
  {
    sys_error("open error");
  }
  ftruncate(fd,100);                // 扩展空间大小
  int len = lseek(fd,0,SEEK_END);
  p = mmap(NULL, len, PROT_READ|PROT_WRITE, MAP_SHARED,fd, 0);    // 创建共享映射区
  if(p == MAP_FAILED)
  {
    sys_error("map error!");
  }
  close(fd);
  pid = fork();                // 创建子进程
  if(pid == 0)
  {
    strcpy(p,"This is child");
    var = 100;
    printf("%s, Child: var = %d\n",p, var);
  }
  else
  {
    sleep(1);
    printf("Parent: %s,var = %d\n",p,var);
  }
  wait(NULL);                     // 回收子进程
  munmap(p, len);                   // 释放映射区
  return 0;
}

var 是 全局变量,父子进程操作 全局变量时,读数据时 共享; 写数据时 复制。

上述代码中,子进程写数据时,是复制一份数据 后 对复制的数据进程修改。父进程 读数据时,全局变量还是原本的数值。

四、非血缘关系进程间 mmap 提通信

1.c 不断写数据:

void sys_error(const char *str)
{
  perror(str);    
  exit(1);
}
int main(void)
{
  int fd;
  char *p;
  int i = 0;
  fd = open("1.txt", O_RDWR);
  if(fd < 0)
  {
    sys_error("open error");
  }
  ftruncate(fd,100);
  int len = lseek(fd,0,SEEK_END);
  p = mmap(NULL, len, PROT_READ|PROT_WRITE, MAP_SHARED,fd, 0);
  if(p == MAP_FAILED)
  {
    sys_error("map error!");
  }
  close(fd);
  while(1)
  {
    sleep(1);
    *p = i;               // 不断写入数据
    i++;
  }
  munmap(p, len);
  return 0;
}

2.c 不断读数据

void sys_error(const char *str)
{
  perror(str);    
  exit(1);
}
int main(void)
{
  int fd;
  char *p;
  int i = 0;
  fd = open("1.txt", O_RDWR);
  if(fd < 0)
  {
    sys_error("open error");
  }
  ftruncate(fd,100);
  int len = lseek(fd,0,SEEK_END);
  p = mmap(NULL, len, PROT_READ|PROT_WRITE, MAP_SHARED,fd, 0);
  if(p == MAP_FAILED)
  {
    sys_error("map error!");
  }
  close(fd);
  while(1)
  {
    sleep(1);
    printf("*p = %d\n",*p);       // 不断读数据
  }
  munmap(p, len);
  return 0;
}

五、mmap 匿名映射区

mmap 匿名映射区是在进程的虚拟内存空间中创建的一段没有对应物理文件的内存区域。它通常用于进程间通信和临时存储数据,不需要使用文件作为映射源。匿名映射区在 Linux 系统中非常常见。


在使用mmap系统调用创建匿名映射区时,传递给mmap函数的文件描述符参数(通常为-1)表明不会有一个与之相关联的文件。

mmap 函数的 参数二,可以为指定的大小。参数四 为 MAP_SHARED|MAP_ANONYMOUS

void sys_error(const char *str)
{
  perror(str);    
  exit(1);                  // 正常退出程序
}
int var = 10;
int main(void)
{
  char *p;
  pid_t pid;
  p = mmap(NULL, 20, PROT_READ|PROT_WRITE, MAP_SHARED|MAP_ANONYMOUS, -1, 0);    // 创建共享映射区
  if(p == MAP_FAILED)
  {
    sys_error("map error!");
  }
  pid = fork();                // 创建子进程
  if(pid == 0)
  {
    strcpy(p,"This is child");
    var = 100;
    printf("%s, Child: var = %d\n",p, var);
  }
  else
  {
    sleep(1);
    printf("Parent: %s,var = %d\n",p,var);
  }
  wait(NULL);                     // 回收子进程
  munmap(p, 20);                    // 释放映射区
  return 0;
}

总结

进程间共享内存映射(mmap)通信是一种高效、灵活的进程间通信机制。通过内存映射,不同的进程可以共享相同的数据区域,提高数据访问速度和性能。然而,在使用该机制时需要注意同步机制、内存管理和安全性等问题,以确保共享数据的正确性和安全性。

相关文章
|
2月前
|
安全 Go 开发者
“不要通过共享内存来通信”——深入理解Golang并发模型与CSP理论
Golang 采用 CSP 理念,主张“通过通信共享内存”,以消息传递替代共享内存,避免数据竞争。其核心为 Goroutine 与 Channel:轻量协程并发执行,通道安全传递数据,将并发复杂性转为通信编排,提升程序清晰度与可维护性。
203 0
|
7月前
|
Web App开发 Linux 程序员
获取和理解Linux进程以及其PID的基础知识。
总的来说,理解Linux进程及其PID需要我们明白,进程就如同汽车,负责执行任务,而PID则是独特的车牌号,为我们提供了管理的便利。知道这个,我们就可以更好地理解和操作Linux系统,甚至通过对进程的有效管理,让系统运行得更加顺畅。
215 16
|
6月前
|
监控 Shell Linux
Linux进程控制(详细讲解)
进程等待是系统通过调用特定的接口(如waitwaitpid)来实现的。来进行对子进程状态检测与回收的功能。
131 0
|
6月前
|
存储 负载均衡 算法
Linux2.6内核进程调度队列
本篇文章是Linux进程系列中的最后一篇文章,本来是想放在上一篇文章的结尾的,但是想了想还是单独写一篇文章吧,虽然说这部分内容是比较难的,所有一般来说是简单的提及带过的,但是为了让大家对进程有更深的理解与认识,还是看了一些别人的文章,然后学习了学习,然后对此做了总结,尽可能详细的介绍明白。最后推荐一篇文章Linux的进程优先级 NI 和 PR - 简书。
200 0
|
6月前
|
存储 Linux Shell
Linux进程概念-详细版(二)
在Linux进程概念-详细版(一)中我们解释了什么是进程,以及进程的各种状态,已经对进程有了一定的认识,那么这篇文章将会继续补全上篇文章剩余没有说到的,进程优先级,环境变量,程序地址空间,进程地址空间,以及调度队列。
133 0
|
6月前
|
Linux 调度 C语言
Linux进程概念-详细版(一)
子进程与父进程代码共享,其子进程直接用父进程的代码,其自己本身无代码,所以子进程无法改动代码,平时所说的修改是修改的数据。为什么要创建子进程:为了让其父子进程执行不同的代码块。子进程的数据相对于父进程是会进行写时拷贝(COW)。
179 0
|
3月前
|
Linux 应用服务中间件 Shell
二、Linux文本处理与文件操作核心命令
熟悉了Linux的基本“行走”后,就该拿起真正的“工具”干活了。用grep这个“放大镜”在文件里搜索内容,用find这个“探测器”在系统中寻找文件,再用tar把东西打包带走。最关键的是要学会使用管道符|,它像一条流水线,能把这些命令串联起来,让简单工具组合出强大的功能,比如 ps -ef | grep 'nginx' 就能快速找出nginx进程。
二、Linux文本处理与文件操作核心命令
|
3月前
|
Linux
linux命令—stat
`stat` 是 Linux 系统中用于查看文件或文件系统详细状态信息的命令。相比 `ls -l`,它提供更全面的信息,包括文件大小、权限、所有者、时间戳(最后访问、修改、状态变更时间)、inode 号、设备信息等。其常用选项包括 `-f` 查看文件系统状态、`-t` 以简洁格式输出、`-L` 跟踪符号链接,以及 `-c` 或 `--format` 自定义输出格式。通过这些选项,用户可以灵活获取所需信息,适用于系统调试、权限检查、磁盘管理等场景。
317 137
|
3月前
|
安全 Ubuntu Unix
一、初识 Linux 与基本命令
玩转Linux命令行,就像探索一座新城市。首先要熟悉它的“地图”,也就是/根目录下/etc(放配置)、/home(住家)这些核心区域。然后掌握几个“生存口令”:用ls看周围,cd去别处,mkdir建新房,cp/mv搬东西,再用cat或tail看文件内容。最后,别忘了随时按Tab键,它能帮你自动补全命令和路径,是提高效率的第一神器。
724 57
|
2月前
|
存储 安全 Linux
Linux卡在emergency mode怎么办?xfs_repair 命令轻松解决
Linux虚拟机遇紧急模式?别慌!多因磁盘挂载失败。本文教你通过日志定位问题,用`xfs_repair`等工具修复文件系统,三步快速恢复。掌握查日志、修磁盘、验重启,轻松应对紧急模式,保障系统稳定运行。
490 2

热门文章

最新文章