ICode9

精准搜索请尝试: 精确搜索
首页 > 编程语言> 文章详细

Java IO学习笔记三:MMAP与RandomAccessFile

2021-06-12 17:34:27  阅读:283  来源: 互联网

标签:map Java 映射 文件 MMAP FileChannel RandomAccessFile size


作者:Grey

原文地址:Java IO学习笔记三:MMAP与RandomAccessFile

关于RandomAccessFile

相较于前面提到的BufferedReader/Writer和FileReader/Writer

普通的Reader和Writer只能顺序读写数据,RandomAccessFile提供了一个独有的seek方法,可以修改文件内容的指针,从而可以方便读取和修改文件中的任意位置。示例:

import java.io.RandomAccessFile;
import java.nio.MappedByteBuffer;
import java.nio.channels.FileChannel;

import static java.nio.charset.StandardCharsets.UTF_8;

public class TestRandomAccessFile {

    public static void main(String[] args) {
        String path = "C:\\workspace\\out.txt";
        try {
            RandomAccessFile randomAccessFile = new RandomAccessFile(path, "rw");
            randomAccessFile.write("Hello xxxld".getBytes(UTF_8));
            randomAccessFile.seek(6);
            randomAccessFile.write("Wor".getBytes(UTF_8));
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
}

seek(6) 让指针跳到"Hello xxld"这个字符串中的第一个"x"的位置,再次写入"Wor"这个字符串(覆盖写)

所以返回的结果是:

Hello World

关于MMAP

关于mmap,可以参考这篇文章的介绍认真分析mmap:是什么 为什么 怎么用

mmap是一种内存映射文件的方法,即将一个文件或者其它对象映射到进程的地址空间,实现文件磁盘地址和进程虚拟地址空间中一段虚拟地址的一一对映关系。实现这样的映射关系后,进程就可以采用指针的方式读写操作这一段内存,而系统会自动回写脏页面到对应的文件磁盘上,即完成了对文件的操作而不必再调用read,write等系统调用函数。相反,内核空间对这段区域的修改也直接反映用户空间,从而可以实现不同进程间的文件共享。

以及从内核文件系统看文件读写过程

通过RandomAccessFile可以拿到文件的FileChannel,并做内存映射(底层就是MMAP,因为文件是块设备,可以来回自由寻址,所以只有FileChannel才能做内存映射),即:把内核的pagecache和文件的数据地址空间映射起来。


    public static void testRandomAccessFileIO() throws Exception {
        String path = "C:\\workspace\\out.txt";
        byte[] data = "1234567\n".getBytes();
        RandomAccessFile file = new RandomAccessFile(path, "rw");
        FileChannel channel = file.getChannel();
        int size = 4096 * 100;
        MappedByteBuffer map = channel.map(FileChannel.MapMode.READ_WRITE, 0, size);
        while (size != 0) {
            map.put(data);
            size -= data.length;
        }
        map.force();
    }

其中

 channel.map(FileChannel.MapMode.READ_WRITE, 0, size);

通过mmap生成的且是堆外的和文件映射的内存区域

原先我们的FileWriter需要通过write方法才能将数据写入pagecache,现在只需要通过map.put方法即可。

其中

map.force()

方法就等同于File中的

file.flush()

方法

将内容从主存写入磁盘中。

源码

Github

标签:map,Java,映射,文件,MMAP,FileChannel,RandomAccessFile,size
来源: https://www.cnblogs.com/greyzeng/p/14878108.html

本站声明: 1. iCode9 技术分享网(下文简称本站)提供的所有内容,仅供技术学习、探讨和分享;
2. 关于本站的所有留言、评论、转载及引用,纯属内容发起人的个人观点,与本站观点和立场无关;
3. 关于本站的所有言论和文字,纯属内容发起人的个人观点,与本站观点和立场无关;
4. 本站文章均是网友提供,不完全保证技术分享内容的完整性、准确性、时效性、风险性和版权归属;如您发现该文章侵犯了您的权益,可联系我们第一时间进行删除;
5. 本站为非盈利性的个人网站,所有内容不会用来进行牟利,也不会利用任何形式的广告来间接获益,纯粹是为了广大技术爱好者提供技术内容和技术思想的分享性交流网站。

专注分享技术,共同学习,共同进步。侵权联系[81616952@qq.com]

Copyright (C)ICode9.com, All Rights Reserved.

ICode9版权所有