起初

看过一遍内存映射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. 共享内存、网络(day13)

    一.共享内存 .获取一个键值 ftok() .使用键值获取共享内存的id shmget() #include <sys/ipc.h> #include <sys/shm.h> ...

  2. 环境变量、system(day10)

    一.环境变量 bash下的环境变量. 每个进程都默认从父进程继承环境变量 bash本身就是一个程序,这个程序运行的时候,bash进程 可以定义只能之自己这个进程中使用的变量,这种变量称为自定义变量. ...

  3. [luogu1155 NOIP2008] 双栈排序 (二分图染色)

    传送门 Description Input 第一行是一个整数 n . 第二行有 n 个用空格隔开的正整数,构成一个 1−n 的排列. Output 共一行,如果输入的排列不是"可双栈排序排列 ...

  4. (35)Spring Boot集成Redis实现缓存机制【从零开始学Spring Boot】

    [本文章是否对你有用以及是否有好的建议,请留言] 本文章牵涉到的技术点比较多:Spring Data JPA.Redis.Spring MVC,Spirng Cache,所以在看这篇文章的时候,需要对 ...

  5. redis实现分页技术

    声明:原博客在这里https://www.cnblogs.com/find-the-right-direction/p/8465011.html,谢谢哥们提供,尊重原创. 本人是在原有的springb ...

  6. 【ACM】NYOJ_506_洗澡_20130725

    洗澡时间限制:1000 ms  |  内存限制:65535 KB 难度:1描述 Mostrp是个爱干净的好少年. 有一次去澡堂洗澡时发现 澡堂的澡柜编号中没有出现过数字‘4’. Mostrp 感到很好 ...

  7. 【Cocos2dx】资源目录,播放背景音乐,导入外部库

    在Cocos2dx中播放背景音乐是一件非常easy的事情,就一行代码,可是首先要导入Cocos2dx的音频引擎cocosDenshion. cocosDenshion对cocos2dproject提供 ...

  8. 深入浅出游戏算法(4)-unity3d算法(1)-球转动

    球 转动 按以下布局放置好unity3d的各个组件.设置好渲染.位置.光源.大小等 麦好的AI乐园博客全部内容是原创,假设转载请注明来源 http://blog.csdn.net/myhaspl/ 编 ...

  9. 手游server之数据IO进化

    这里数据IO是指游戏数据存盘和读取. 假设IO处理不好.server在IO时会导致.游戏卡顿较长的时间,严重影响游戏体验. 近期服务端刚好对IO这一块做了优化,把优化过程记录一下. 一 原始版 刚開始 ...

  10. luogu2437 蜜蜂路线

    题目大意 一只蜜蜂在下图所示的数字蜂房上爬动,已知它只能从标号小的蜂房爬到标号大的相邻蜂房,现在问你:蜜蜂从蜂房M开始爬到蜂房N,M<N,有多少种爬行路线?M,N<=1000 题解 看到M ...