分享
 
 
 

java nio 之MappedByteBuffer

王朝学院·作者佚名  2009-11-21
窄屏简体版  字體: |||超大  

其实掌握 MappedByteBuffer 并不难,只要记住“三方三法三特性”(我自己总结的,呵呵 ~~ 不要扔鸡蛋哦。。。)这句话就可以轻松搞定! MappedByteBuffer 只是一种特殊的 ByteBuffer ,即是 ByteBuffer 的子类。 MappedByteBuffer 将文件直接映射到内存(这里的内存指的是虚拟内存,并不是物理内存,后面说证明这一点)。通常,可以映射整个文件,如果文件比较大的话可以分段进行映射,只要指定文件的那个部分就可以。而且,与 ByteBuffer 十分类似,没有构造函数(你不可 new MappedByteBuffer ()来构造一个 MappedByteBuffer ),我们可以通过 java.nio.channels.FileChannel 的 map() 方法来获取 MappedByteBuffer 。其实说的通俗一点就是 Map 把文件的内容被映像到计算机虚拟内存的一块区域,这样就可以直接操作内存当中的数据而无需操作的时候每次都通过 I/O 去物理硬盘读取文件,所以效率上有很大的提升!

三种方式:

FileChannel 提供了 map 方法来把文件影射为内存映像文件: MappedByteBuffer map(int mode,long position,long size); 可以把文件的从 position 开始的 size 大小的区域映射为内存映像文件, mode 指出了 可访问该内存映像文件的方式: READ_ONLY,READ_WRITE,PRIVATE.

a. READ_ONLY, (只读): 试图修改得到的缓冲区将导致抛出 ReadOnlyBufferException .(MapMode.READ_ONLY )

b. READ_WRITE (读 / 写): 对得到的缓冲区的更改最终将传播到文件;该更改对映射到同一文件的其他程序不一定是可见的。 ( MapMode.READ_WRITE )

c. PRIVATE (专用): 对得到的缓冲区的更改不会传播到文件,并且该更改对映射到同一文件的其他程序也不是可见的;相反,会创建缓冲区已修改部分的专用副本。 ( MapMode.PRIVATE )

三个方法:

a. fore(); 缓冲区是 READ_WRITE 模式下,此方法对缓冲区内容的修改强行写入文件

b. load() 将缓冲区的内容载入内存,并返回该缓冲区的引用

c. isLoaded() 如果缓冲区的内容在物理内存中,则返回真,否则返回假

三个特性:

调用信道的 map() 方法后,即可将文件的某一部分或全部映射到内存中,映射内存缓冲区是个直接缓冲区,继承自 ByteBuffer, 但相对于 ByteBuffer, 它有更多的优点:

a. 读取快

b. 写入快

c. 随时随地写入

口说无凭,俗话说的好,是金子是银子拿来炼一炼就知道 ( 也不知道有没有这么一句俗话,反正用到这还合适,就这么凑合吧,大家扔鸡蛋 ~~~) !

1 MappedByteBuffer 的读取 / 写入文件和普通 I/O 流的对比

import java.io.FileInputStream;

import java.io.FileOutputStream;

import java.nio.ByteBuffer;

import java.nio.CharBuffer;

import java.nio.MappedByteBuffer;

import java.nio.channels.FileChannel;

import java.nio.charset.Charset;

import java.nio.charset.CharsetDecoder;

public class MapMemeryBuffer {

public static void main(String[] args) throws Exception {

ByteBuffer byteBuf = ByteBuffer.allocate(1024 * 14 * 1024);

byte[] bbb = new byte[14 * 1024 * 1024];

FileInputStream fis = new FileInputStream("d:\\test");

FileOutputStream fos = new FileOutputStream("d:\\outFile.txt");

FileChannel fc = fis.getChannel();

long timeStar = System.currentTimeMillis();// 得到当前的时间

fc.read(byteBuf);//1 读取

long timeEnd = System.currentTimeMillis();// 得到当前的时间

System.out.println("Read time :" + (timeEnd - timeStar) + "ms");

timeStar = System.currentTimeMillis();

fos.write(bbb);// 写入

timeEnd = System.currentTimeMillis();

System.out.println("Write time :" + (timeEnd - timeStar) + "ms");

fos.flush();

fc.close();

fis.close();

}

}

输出结果:

Read time :1874ms

Write time :360ms

把上面的程序的 1 换成 MappedByteBuffer mbb = fc.map(FileChannel.MapMode.READ_ONLY, 0, fileLength);

2 换成 mbb.flip();

输出结果:

Read ByteBuf take time :16ms

Write ByteBuf take time :0ms

可见普通 I/O 和 MappedByteBuffer 是没法比的。另外在写入的时候花了 0ms 说明 Map 写入机制是根据你的更改量来决定,就是只保存修改部分的!

 
 
 
免责声明:本文为网络用户发布,其观点仅代表作者个人观点,与本站无关,本站仅提供信息存储服务。文中陈述内容未经本站证实,其真实性、完整性、及时性本站不作任何保证或承诺,请读者仅作参考,并请自行核实相关内容。
2023年上半年GDP全球前十五强
 百态   2023-10-24
美众议院议长启动对拜登的弹劾调查
 百态   2023-09-13
上海、济南、武汉等多地出现不明坠落物
 探索   2023-09-06
印度或要将国名改为“巴拉特”
 百态   2023-09-06
男子为女友送行,买票不登机被捕
 百态   2023-08-20
手机地震预警功能怎么开?
 干货   2023-08-06
女子4年卖2套房花700多万做美容:不但没变美脸,面部还出现变形
 百态   2023-08-04
住户一楼被水淹 还冲来8头猪
 百态   2023-07-31
女子体内爬出大量瓜子状活虫
 百态   2023-07-25
地球连续35年收到神秘规律性信号,网友:不要回答!
 探索   2023-07-21
全球镓价格本周大涨27%
 探索   2023-07-09
钱都流向了那些不缺钱的人,苦都留给了能吃苦的人
 探索   2023-07-02
倩女手游刀客魅者强控制(强混乱强眩晕强睡眠)和对应控制抗性的关系
 百态   2020-08-20
美国5月9日最新疫情:美国确诊人数突破131万
 百态   2020-05-09
荷兰政府宣布将集体辞职
 干货   2020-04-30
倩女幽魂手游师徒任务情义春秋猜成语答案逍遥观:鹏程万里
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案神机营:射石饮羽
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案昆仑山:拔刀相助
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案天工阁:鬼斧神工
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案丝路古道:单枪匹马
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案镇郊荒野:与虎谋皮
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案镇郊荒野:李代桃僵
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案镇郊荒野:指鹿为马
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案金陵:小鸟依人
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案金陵:千金买邻
 干货   2019-11-12
 
推荐阅读
 
 
 
>>返回首頁<<
 
靜靜地坐在廢墟上,四周的荒凉一望無際,忽然覺得,淒涼也很美
© 2005- 王朝網路 版權所有