起初

看过一遍内存映射I/O,意思大概是懂了,就是直接操作文件再而直接通过缓冲区来操作,减少一些read、write调用所花费的时间。加上文中给出一个copy的例子,意思也好理解的。
不过困扰的来了,我琢磨着在映射两个文件后,再使用memcpy内存复制函数将文件写入完全可以,但是这是两个文件的操作,我想一个文件自己对自己的修改呢?只有一个文件的时候,该怎么做呢?于是自己就开始构建一个程序来验证一下。正好也顺手将书后的习题14.11也解决掉:)


猜想

首先设想创建一个空文件,先写入一点数据,接着映射到缓冲区区,接着再写一点数据进去。简单说就是open==> write ==> mmap ==> write 但是一想这边根本没有涉及到使用缓冲区,还是建立在文件的操作上,失败!

继续想一个思路,书上有一句话 此函数的返回地址是该映射区的返回地址 ,意思就是说mmap的返回值可以拿过来用,我们可以将它的内容输出,我个人猜想这个内容应该就是文件的内容。那么这样我不就可以对这个返回值进行相应的操作,这不就是在对映射的缓存区进行操作吗?

鉴于书中提到一个函数msync是将映射区的修改同步到文件中,那么对文件的修改也就能体现出来了。一气呵成的想法,最后还有一个释放映射区,这是一个基本的清理操作。大体上应该可以设计了。

思路整理下,首先是创建文件,写入一点数据,然后映射过去,读映射区的内容(返回值) 修改映射区,同步写入文件,清理工作。这个思路应该差不多了。


实现

编码之前的一个问题,mmap的返回值是void *, 而我们要展示的确是字符串之类的,于是想到得强制转换下这个返回值。这一步解决后,开始编码

#include<stdio.h>
#include<stdlib.h>
#include<string.h>
#include<fcntl.h>
#include<unistd.h>
#include<sys/mman.h>
#include<errno.h> static void error(const char *msg)
{
fprintf(stderr,"%s:%s\n",msg,strerror(errno));
exit();
} int main(void)
{
int fd;
char *dst; if((fd = open("mapfile",O_RDWR|O_CREAT|O_TRUNC,)) < )
error("open error"); if(write(fd,"abc",) != )
error("wirte error");
/*let this file map to memory area*/
if((dst =(char *)mmap(,,PROT_READ|PROT_WRITE,MAP_SHARED,fd,)) == MAP_FAILED) error("mmap error"); close(fd);
printf("%s\n",dst);
dst[]='d';
printf("%s\n",dst);
if(msync((void *)dst,,MS_SYNC) < )
error("msync error"); if(munmap((void *)dst,) < )
error("munmap error"); return ; }

初期的一个编码大概是这样的,运行后的结果还是比较不错的.

细究下这个代码,我尝试的是去改已经写入的内容,这个确实有一个修改的意思了,看来对返回值的猜想是正确的,对返回值的修改就是对这个映射区的修改。同时,我也测试了将文件描述符关闭后,缓存区是不受影响的,照样可以操作,并且是能够同步到文件中的。这点确实很关键。

可我并不觉着这样就可以了,我的目的是继续往里面写数据,就像一开始想的那样,在映射后我还需要将其他数据写入, 看看这个是否是和write函数一样可以做到这一点。

修改代码

我们一开始只是写入了3个,而给的缓存区是64的,足够再写入的。但是运行的结果

和上面的一样,没有任何的改变。。。 按照道理来时应该会有一个x出现的。

看书,发现书上的copy函数的例子中,有一个偏移的设置,书中也有个说法映射文件的启示偏移量受系统虚存页的限制,如果映射区的长度不是页的整数倍,任何的变动,都不会再文件中反映出来,不能用mmap将数据添加到文件中,需要加长该文件 也就是说我这边需要一个偏移,看书上的copy的例子,上面是从文件开始偏移了inputfile的长度。也就是说现在这个文件从开始到偏移的位置有一个空洞,那么再映射过去后,显示的量应该就是能满足了,这样就可以将所更改的写入到文件中。回到我刚才的那个,虽然映射区是给64,但是文件的长度不足,这和write不一样,write可以直接接着后面写。所以必须文件的大小要满足才行。

继续修改代码

此刻运行的结果就符合我的猜想了,下面是结果的运行

可以看到,cat输出了我刚刚插入的一个'x',继而看文件的真实情况,也完全符合我的猜想,在第十一位放入了一个x,至于为什么输出的还是三个,因为字符串的结尾有一个'\0'则'x'没有连在一起。

其实还有一个细节,再形成文件空洞的时候,不仅仅是要偏移位置,还需要写入一个字符,来标识这个文件真实长度,否则结果和不使用偏移一样文件的长度还是原来的大小,并没有被扩充。

有一个不错的方法使用ftruncate,直接设置文件的大小,省却使用lseek和write的繁杂。


总结

mmap再IPC中提及到,但是在APUE中涉及了这个,这让我想起了再做摄像头驱动的时候,使用过这个函数,但是一点概念一点都没有,完全跟着老师后一个字一个字的敲,现在大概能理解使用存储映射的过程,至于什么二级缓存,一级缓存,感觉需要到实际项目中才能感受到。

(全文完)

<轉>APUE:mmap函数的更多相关文章

  1. mmap函数实现

    转自:https://www.cnblogs.com/huxiao-tee/p/4660352.htmlmmap是一种内存映射文件的方法,即将一个文件或者其它对象映射到进程的地址空间,实现文件磁盘地址 ...

  2. UNIX环境高级编程——存储映射I/O(mmap函数)

         共享内存可以说是最有用的进程间通信方式,也是最快的IPC形式,因为进程可以直接读写内存,而不需要任何数据的拷贝.对于像管道和消息队列等通信方式,则需要在内核和用户空间进行四次的数据拷贝,而共 ...

  3. linux 内存映射-ioremap和mmap函数

    最近开始学习Linux驱动程序,将内存映射和ioremap,mmap函数相关资料进行了整理 一,内存映射  对于提供了MMU(存储管理器,辅助操作系统进行内存管理,提供虚实地址转换等硬件支持)的处理器 ...

  4. 共享内存:mmap函数实现

    内存映射的应用: 以页面为单位,将一个普通文件映射到内存中,通常在须要对文件进行频繁读写时使用,这样用内存读写代替I/O读写,以获得较高的性能; 将特殊文件进行匿名内存映射,能够为关联进程提供共享内存 ...

  5. 共享内存简介和mmap 函数

    一.共享内存简介 共享内存区是最快的IPC形式,这些进程间数据传递不再涉及到内核,换句话说是进程不再通过执行进入内核的系统调用来传递彼此的数据. 即每个进程地址空间都有一个共享存储器的映射区,当这块区 ...

  6. mmap函数使用

    UNIX网络编程第二卷进程间通信对mmap函数进行了说明.该函数主要用途有三个:1.将一个普通文件映射到内存中,通常在需要对文件进行频繁读写时使用,这样用内存读写取代I/O读写,以获得较高的性能:2. ...

  7. 【APUE】进程间通信之共享存储(mmap函数)

    共享内存可以说是最有用的进程间通信方式,也是最快的IPC形式,因为进程可以直接读写内存,而不需要任何数据的拷贝.对于像管道和消息队列等通信方式,则需要在内核和用户空间进行四次的数据拷贝,而共享内存则只 ...

  8. Linux系统编程(5)——文件与IO之mmap函数

    mmap系统调用它本身提供了不同于一般对普通文件的访问方式,进程可以像读写内存一样对普通文件的操作.而Posix或系统V的共享内存IPC则纯粹用于共享目的,mmap()实现共享内存也是其主要应用之一. ...

  9. Linux内存映射--mmap函数

    Linux提供了内存映射函数mmap, 它把文件内容映射到一段内存上(准确说是虚拟内存上), 通过对这段内存的读取和修改, 实现对文件的读取和修改, 先来看一下mmap的函数声明: 头文件: < ...

随机推荐

  1. 日记——OI历程

    学OI也一年多了(2015.12-),一直没学出个像样的东西.相比dalao们,我还是弱爆了. ljj,qyf,yyf三位三区dalao. xxy,myj两位三区学长dalao. 稍微总结一下前一段时 ...

  2. intellij idea 的历史版本

    开发工具intellij idea 的历史版本https://www.jetbrains.com/idea/download/previous.html

  3. 39.exact value and full text

    主要知识点 1.exact value 2.full text     在es中有两种搜索方式,exact value(精确匹配),full text(全文检索) 一.exact value 比如有一 ...

  4. 继续聊WPF——进度条

    ProgressBar控件与传统WinForm使用方法完全一样,我们只需关注: Minimum——最小值,默认为0: Maximum——最大值,默认为100. Value——当前值.   关键是它的控 ...

  5. nutz_web应用中主页跳转到登录页面的方式

    一.前言 web应用开发时,地址栏输入ip+port+appName,通常可以跳转到登录页面.以下便介绍我所知道并且验证过的三种跳转方式. 二.准备工作 需要使用到两个url的处理分别如下: @At( ...

  6. 【j2ee spring】44、巴巴运动网前台产品显示

    [j2ee spring]44.巴巴运动网前台产品显示 项目结构 项目代码 界面显示 <%@ page language="java" isELIgnored="f ...

  7. Linux学习(二) wget命令的使用

    近期在Linux下进行一些操作,在非常多地方都用到了wget这个命令,记录一下一些有关wget的使用方法: wget是在Linux下开发的开放源码的软件,作者是Hrvoje Niksic,后来被移植到 ...

  8. 用BFS解决迷宫问题

    在一个n*n的矩阵里走,从原点(0,0)開始走到终点(n-1,n-1),仅仅能上下左右4个方向走.仅仅能在给定的矩阵里走,求最短步数. n*n是01矩阵,0代表该格子没有障碍.为1表示有障碍物. in ...

  9. Spring如何加载XSD文件(org.xml.sax.SAXParseException: Failed to read schema document错误的解决方法)

    今天配置Spring的xml出现了错误 Multiple annotations found at this line: - schema_reference.4: Failed to read sc ...

  10. 【POJ 2044】 Weather Forecast

    [题目链接] http://poj.org/problem?id=2044 [算法] 广度优先搜索 [代码] #include <algorithm> #include <bitse ...