linux 进程间通信——内存共享映射mmap和munmap
mmap可以把磁盘文件的一部分直接映射到内存,这样文件中的位置直接就有对应的内存地址,对文件的读写可以直接用指针来操做而不需要read/write函数。
#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);
  如果addr参数为NULL,内核会自己在进程地址空间中选择合适的地址建立映射。如果addr不是NULL,则给内核一个提示,应该从什么地址开始映射,内核会选择addr之上的某个合适的地址开始映射。建立映射后,真正的映射首地址通过返回值可以得到。len参数是需要映射的那一部分文件的长度。off参数是从文件的什么位置开始映射,必须是页大小的整数倍(在32位体系统结构上通常是4K)。filedes是代表该文件的描述符。 
   
prot参数有四种取值: 
  1)PROT_EXEC 表示映射的这一段可执行,例如映射共享库 
  2)PROT_READ 表示映射的这一段可读 
  3)PROT_WRITE 表示映射的这一段可写 
  4) PROT_NONE 表示映射的这一段不可访问 
    
flag参数有很多种取值,这里只讲两种常用的,其它取值可查看mmap(2)
- MAP_SHARED 多个进程对同一个文件的映射是共享的,一个进程对映射的内存做了修改,另一个进程也会看到这种变化。
- MAP_PRIVATE 多个进程对同一个文件的映射不是共享的,一个进程对映射的内存做了修改,另一个进程并不会看到这种变化,也不会真的写到文件中去。
* 如果mmap成功则返回映射首地址,如果出错则返回常数MAP_FAILED。当进程终止时,该进程的映射内存会自动解除,也可以调用munmap解除映射。munmap成功返回0,出错返回-1。

/* 运行前准备 */
book@ubuntu:~$ vi hello
book@ubuntu:~$ cat hello
helloworld
book@ubuntu:~$ od -tx1 -tc hello
0000000 68 65 6c 6c 6f 77 6f 72 6c 64 0a
h e l l o w o r l d \n


#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <sys/stat.h>
#include <sys/types.h>
#include <fcntl.h>
#include <sys/mman.h> int main(void)
{
int fd, *p, len;
fd = open("hello",O_RDWR);
if(fd < 0)
{
perror("open");
exit(1);
}
//获取文件长度
len = lseek(fd, 0, SEEK_END);
//建立映射
p = mmap(NULL, len, PROT_READ|PROT_WRITE, MAP_SHARED, fd, 0);
if(p == MAP_FAILED)
{//如果失败,记住这种检测方式
perror("mmap");
exit(1);
}
//即使关闭文件也不会释放映射
close(fd);
//由于是共享映射,所以映射源也会被修改
p[0] = 0x30313233; munmap(p, len); return 0;
} /* 运行后结果(小端存储的原因) */
book@ubuntu:~$ cat hello
3210oworld
book@ubuntu:~$ od -tx1 -tc hello
0000000 33 32 31 30 6f 77 6f 72 6c 64 0a
3 2 1 0 o w o r l d \n

利用内存映射的特性,也可以用于进程间的通信,但是要注意的是:
1、用于进程间通信时,一般设计成结构体,来传输通信的数据
2、进程间通信的文件,应该设计成临时文件(即创建文件,使用文件,删除文件)
3、 当报总线错误时,优先查看共享文件是否有存储空间
linux 进程间通信——内存共享映射mmap和munmap的更多相关文章
- Linux进程间通信—使用共享内存
		Linux进程间通信-使用共享内存 转自: https://blog.csdn.net/ljianhui/article/details/10253345 下面将讲解进程间通信的另一种方式,使用共享内 ... 
- linux进程间通信之共享内存篇
		本文是对http://www.cnblogs.com/andtt/articles/2136279.html中共享内存(上)的进一步阐释说说明 1 共享内存的实现原理 共享内存是linux进程间通讯的 ... 
- Linux进程间通信(四) - 共享内存
		共享内存的优势 采用共享内存通信的一个显而易见的好处是效率高,因为进程可以直接读写内存,而不需要任何数据的拷贝.对于像管道和消息队列等通信方式,则需要在内核和用户空间进行四次的数据拷贝,而共享内存则只 ... 
- Linux进程间通信——使用共享内存
		一.什么是共享内存 顾名思义,共享内存就是允许两个不相关的进程访问同一个逻辑内存.共享内存是在两个正在运行的进程之间共享和传递数据的一种非常有效的方式.不同进程之间共享的内存通常安排为同一段物理内存. ... 
- Linux进程间通信——使用共享内存(转)
		一.什么是共享内存 顾名思义,共享内存就是允许两个不相关的进程访问同一个逻辑内存.共享内存是在两个正在运行的进程之间共享和传递数据的一种非常有效的方式.不同进程之间共享的内存通常安排为同一段物理内存. ... 
- Linux进程间通信之共享内存
		一,共享内存 内核管理一片物理内存,允许不同的进程同时映射,多个进程可以映射同一块内存,被多个进程同时映射的物理内存,即共享内存. 映射物理内存叫挂接,用完以后解除映射叫脱接. 1,共享内存的特点 ... 
- 【APUE】进程间通信之共享存储(mmap函数)
		共享内存可以说是最有用的进程间通信方式,也是最快的IPC形式,因为进程可以直接读写内存,而不需要任何数据的拷贝.对于像管道和消息队列等通信方式,则需要在内核和用户空间进行四次的数据拷贝,而共享内存则只 ... 
- linux进程间通信之共享内存学习记录
		进程 狭义定义:进程是正在运行的程序的实例(an instance of a computer program that is being executed). 广义定义:进程是一个具有一定独立功能的 ... 
- linux进程间通信同步-共享内存
		参考:https://www.cnblogs.com/charlesblc/p/6142868.html 使用有名信号量,sem_open().sem_close().sem_post().sem_w ... 
随机推荐
- 【MySQL (六) | 详细分析MySQL事务日志redo log】
			Reference: https://www.cnblogs.com/f-ck-need-u/archive/2018/05/08/9010872.html 引言 为了最大程度避免数据写入时 IO ... 
- extjs ajax 同步 及 confirm 确认提示框问题
			//上传文件 uploadModel: function() { if(Ext.getCmp('exup').getForm().isValid()) { var ssn = this.upPanel ... 
- R语言与机器学习学习笔记
			人工神经网络(ANN),简称神经网络,是一种模仿生物神经网络的结构和功能的数学模型或计算模型.神经网络由大量的人工神经元联结进行计算.大多数情况下人工神经网络能在外界信息的基础上改变内部结构,是一种自 ... 
- Excel数据与DateTable数据的转换
			public class ExcelHelper : IDisposable { private string fileName = null; //文件名 private IWorkbook wor ... 
- LostRoutes项目日志——玩家飞机精灵Fighter解析
			Fighter类的定义在Fighter.js中,Fighter类继承与PhysicsSprite. 原版的Fighter.js: var Fighter = cc.PhysicsSprite.exte ... 
- 转载>>C# Invoke和BeginInvoke区别和使用场景
			转载>>C# Invoke和BeginInvoke区别和使用场景 一.为什么Control类提供了Invoke和BeginInvoke机制? 关于这个问题的最主要的原因已经是dotnet程 ... 
- Mac - 如何安全地还原 Mac 的默认字体
			为清理mac系统内存,下载了daisydisk for mac 破解版,然后发现mac所有的字体都被清理了,所有汉子都变成了问号❓和方框.... 在通常情况下,遇到字体显示不正常,甚至乱码时,重置总是 ... 
- 11.13git和redis
			2018-11-13 15:46:40 git完结 redis完结, 还剩一点路飞项目!!!做完回学校!!!! 越努力,越幸运!永远不要高估自己! 关于 redis具体使用可以参考: http://w ... 
- poj3723_Conscription
			Conscription Time Limit: 1000MS Memory Limit: 65536K Total Submissions: 12393 Accepted: 4350 Des ... 
- hdu 4544——消灭兔子
			游戏规则很简单,用箭杀死免子即可. 箭是一种消耗品,已知有M种不同类型的箭可以选择,并且每种箭都会对兔子造成伤害,对应的伤害值分别为Di(1 <= i <= M),每种箭需要一定的QQ币 ... 
