首页 > 系统相关 >Linux 共享内存mmap,进程通信

Linux 共享内存mmap,进程通信

时间:2023-08-12 20:31:35浏览次数:37  
标签:映射 int mmap void error fd Linux 共享内存

@TOC

前言

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


一、存储映射 I/O

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

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

Linux 共享内存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 :

返回值:

Linux 共享内存mmap,进程通信_共享内存_02

  • 成功 : 映射区的首地址。
  • 失败 : 返回 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;
}

Linux 共享内存mmap,进程通信_共享内存_03

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;
}

Linux 共享内存mmap,进程通信_共享内存_04

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

标签:映射,int,mmap,void,error,fd,Linux,共享内存
From: https://blog.51cto.com/u_16159289/7061544

相关文章

  • 嵌入式Linux dhcp自动配置usb虚拟网卡ip跟主机通信
    dhcpd自动配置usb虚拟网卡ip,与PC机通信配置buildroot勾选dhcpserver修改设备/etc/dhcp/dhcpd.confoptiondomain-name"example.org";optiondomain-name-serversns1.example.org,ns2.example.org;default-lease-time600;max-lease-time7200;ddns-update-stylen......
  • VNC连接Linux图形化界面
    作者:张启昊邮箱:[email protected]更新日期:2023年08月12号Linux安装workstation软件包,其中包含GNOME图形化界面yumgroupinstall-yworkstation开启默认图形化界面systemctlset-defaultgraphical.targetsystemctlisolategraphical.target下载tigervnc......
  • Linux系统之Team链路聚合配置
    (Linux系统之Team链路聚合配置)一、Team链路聚合介绍1.Teaming技术简介1.Teaming技术就是把同一台服务器上的多个物理网卡通过软件绑定成一个虚拟网卡;2.我们可以通过Teaming技术做链路聚合,实现不同网卡的网卡的负载均衡和冗余;2.网卡的bonding和Teaming技术1.一般在在RHE......
  • linux查看内存
    linux查看内存多大的方法:1、执行“free-m”命令,输出列表的“total”项值就是查询的总内存数;2、执行“cat/proc/meminfo”命令,可显示内存的详细信息,输出列表的“MemTotal”项值就是查询的总内存大小。本教程操作环境:linux5.9.8系统、DellG3电脑。linux查看内存多大的方法1......
  • linux中常用端口查询命令
    1、lsof-i:80 用于查看某一端口的占用情况2、netstat-tunlp|grep80 用于查看指定的端口号的进程情况......
  • Linux下使用GDB调试代码
    目录Linux下使用GDB调试代码GDB例程启动查看断点运行退出Linux下使用GDB调试代码GDBGDB是GNU开源组织发布的一个强大的Unix/Linux下的程序调试工具。作用:1、启动用户程序后,可以按照用户的要求随意运行程序2、可让被调试的程序在用户所设定的断点处停住3、当程序被停住时,可......
  • Linux visudo权限管理
    root用户给普通用户授权visudo(推荐使用,因为他会检查语法)vim/etc/suoders(不检查语法)普通用户切换root用户的两种方法1)su-root(需要root密码)2)sudosu-root(不需要root密码但是需要在visudo或/etcsuoders里面配置guoguo ALL =(ALL) ALL 用户    主机 切换的角色 ......
  • Linux ROOT密码忘记解决方法 root口令忘记解决方法
    忘记root密码解决思路:用光盘启动重新设置密码将光盘设置为第一启动保存退出进入救援模式  用光盘启动 设置root密码主板上有个bios芯片,不但可以自检程序用于引导之外,还可以设置(一般电脑的话开机按F2、F1或者其他键)虚拟机上就是打开电源时进入固件然后开机 找到Boot(启动)里面......
  • Linux磁盘故障,模拟故障及解决思路方法
    每个分区起始位置都有一个inod表索引节点表(类似于目录表)每一个文件都对应一个编号称为索引节点,如果这个空间文件数太多了,记满了,就说明索引节点表耗尽。故障1 该分区不能正常读写或者说只能读不能写了但是又没有满,就代表文件系统有问题,文件系统有问题需要进行修复命令:故障2:索引......
  • 解决linux mysql命令 bash mysql command not found 的方法
    错误:root@DB-02~]#mysql-uroot-bash:mysql:commandnotfound原因:这是由于系统默认会查找/usr/bin下的命令,如果这个命令不在这个目录下,当然会找不到命令,我们需要做的就是映射一个链接到/usr/bin目录下,相当于建立一个链接文件。首先得知道mysql命令或mysqladmin命令的完整路......