Java IO读写大文件的几种方式及测试
来源:互联网 发布:知乎员工数量 编辑:程序博客网 时间:2024/04/30 04:28
读取文件大小:1.45G
第一种,OldIO:
耗时70.79s
第二种,newIO:
耗时47.24s
第三种,RandomAccessFile:
耗时46.65
第四种,MappedByteBuffer:
耗时:36
前三种读法对应的资源占用图如下:
相对于最后一种内存直接映射方式前面的测试其实无意义,基本秒杀。。。。。
对于很大的文件直接分块映射时内存会不够,这是因为MappedByteBuffer未被释放造成的,sun未提供直接回收MappedByteBuffer区域的方法,这个时候有两种方法解决,第一种比较愚笨的:
第二种网上找来的,利用反射调用clean方法:
以上两种方法感觉都别扭,还有就是可以自己分割成物理文件再循环调用,这个也不太美观。
速度也会减慢好多。
第一种,OldIO:
- public staticvoid oldIOReadFile() throws IOException{
- BufferedReader br = new BufferedReader(new FileReader("G://lily_947.txt"));
- PrintWriter pw = new PrintWriter("G://oldIO.tmp");
- char[] c = newchar[100*1024*1024];
- for(;;){
- if(br.read(c)!=-1){
- pw.print(c);
- }else{
- break;
- }
- }
- pw.close();
- br.close();
- }
public static void oldIOReadFile() throws IOException{BufferedReader br = new BufferedReader(new FileReader("G://lily_947.txt"));PrintWriter pw = new PrintWriter("G://oldIO.tmp");char[] c = new char[100*1024*1024];for(;;){if(br.read(c)!=-1){pw.print(c);}else{break;}}pw.close();br.close();}
耗时70.79s
第二种,newIO:
- public staticvoid newIOReadFile() throws IOException{
- FileChannel read = new RandomAccessFile("G://lily_947.txt","r").getChannel();
- FileChannel writer = new RandomAccessFile("G://newIO.tmp","rw").getChannel();
- ByteBuffer bb = ByteBuffer.allocate(200*1024*1024);
- while(read.read(bb)!=-1){
- bb.flip();
- writer.write(bb);
- bb.clear();
- }
- read.close();
- writer.close();
- }
public static void newIOReadFile() throws IOException{FileChannel read = new RandomAccessFile("G://lily_947.txt","r").getChannel();FileChannel writer = new RandomAccessFile("G://newIO.tmp","rw").getChannel();ByteBuffer bb = ByteBuffer.allocate(200*1024*1024);while(read.read(bb)!=-1){bb.flip();writer.write(bb);bb.clear();}read.close();writer.close();}
耗时47.24s
第三种,RandomAccessFile:
- public staticvoid randomReadFile() throws IOException{
- RandomAccessFile read = new RandomAccessFile("G://lily_947.txt","r");
- RandomAccessFile writer = new RandomAccessFile("G://random.tmp","rw");
- byte[] b = newbyte[200*1024*1024];
- while(read.read(b)!=-1){
- writer.write(b);
- }
- writer.close();
- read.close();
- }
public static void randomReadFile() throws IOException{RandomAccessFile read = new RandomAccessFile("G://lily_947.txt","r");RandomAccessFile writer = new RandomAccessFile("G://random.tmp","rw");byte[] b = new byte[200*1024*1024];while(read.read(b)!=-1){writer.write(b);}writer.close();read.close();}
耗时46.65
第四种,MappedByteBuffer:
- public staticvoid mappedBuffer() throws IOException{
- FileChannel read = new FileInputStream("G://lily_947.txt").getChannel();
- FileChannel writer = new RandomAccessFile("G://buffer.tmp","rw").getChannel();
- long i = 0;
- long size = read.size()/30;
- ByteBuffer bb,cc = null;
- while(i<read.size()&&(read.size()-i)>size){
- bb = read.map(FileChannel.MapMode.READ_ONLY, i, size);
- cc = writer.map(FileChannel.MapMode.READ_WRITE, i, size);
- cc.put(bb);
- i+=size;
- bb.clear();
- cc.clear();
- }
- bb = read.map(FileChannel.MapMode.READ_ONLY, i, read.size()-i);
- cc.put(bb);
- bb.clear();
- cc.clear();
- read.close();
- writer.close();
- }
public static void mappedBuffer() throws IOException{FileChannel read = new FileInputStream("G://lily_947.txt").getChannel();FileChannel writer = new RandomAccessFile("G://buffer.tmp","rw").getChannel();long i = 0;long size = read.size()/30;ByteBuffer bb,cc = null;while(i<read.size()&&(read.size()-i)>size){bb = read.map(FileChannel.MapMode.READ_ONLY, i, size);cc = writer.map(FileChannel.MapMode.READ_WRITE, i, size);cc.put(bb);i+=size;bb.clear();cc.clear();}bb = read.map(FileChannel.MapMode.READ_ONLY, i, read.size()-i);cc.put(bb);bb.clear();cc.clear();read.close();writer.close();}
耗时:36
前三种读法对应的资源占用图如下:
相对于最后一种内存直接映射方式前面的测试其实无意义,基本秒杀。。。。。
对于很大的文件直接分块映射时内存会不够,这是因为MappedByteBuffer未被释放造成的,sun未提供直接回收MappedByteBuffer区域的方法,这个时候有两种方法解决,第一种比较愚笨的:
- System.gc();
- System.runFinalization();
- try {
- Thread.sleep(3000);
- } catch (InterruptedException e) {
- e.printStackTrace();
- }
System.gc(); System.runFinalization(); try {Thread.sleep(3000);} catch (InterruptedException e) {e.printStackTrace();}
第二种网上找来的,利用反射调用clean方法:
- public staticvoid unmap(final MappedByteBuffer buffer) {
- if (buffer == null) {
- return;
- }
- AccessController.doPrivileged(new PrivilegedAction<Object>() {
- public Object run() {
- try {
- Method getCleanerMethod = buffer.getClass().getMethod("cleaner",new Class[0]);
- if (getCleanerMethod != null) {
- getCleanerMethod.setAccessible(true);
- Object cleaner = getCleanerMethod.invoke(buffer, new Object[0]);
- Method cleanMethod = cleaner.getClass().getMethod("clean",new Class[0]);
- if (cleanMethod != null) {
- cleanMethod.invoke(cleaner, new Object[0]);
- }
- }
- } catch (Exception e) {
- e.printStackTrace();
- }
- return null;
- }
- });
- }
public static void unmap(final MappedByteBuffer buffer) {if (buffer == null) {return;}AccessController.doPrivileged(new PrivilegedAction<Object>() {public Object run() {try {Method getCleanerMethod = buffer.getClass().getMethod("cleaner", new Class[0]);if (getCleanerMethod != null) {getCleanerMethod.setAccessible(true);Object cleaner = getCleanerMethod.invoke(buffer, new Object[0]);Method cleanMethod = cleaner.getClass().getMethod("clean", new Class[0]);if (cleanMethod != null) {cleanMethod.invoke(cleaner, new Object[0]);}}} catch (Exception e) {e.printStackTrace();}return null;} });}
以上两种方法感觉都别扭,还有就是可以自己分割成物理文件再循环调用,这个也不太美观。
速度也会减慢好多。
- Java IO读写大文件的几种方式及测试
- Java IO读写大文件的几种方式及测试
- Java IO读写大文件的几种方式及测试
- Java IO读写大文件的几种方式及测试
- Java IO读写大文件的几种方式及测试
- Java IO读写大文件的几种方式及测试
- java IO读写大文件的几种方式及测试
- Java IO读写大文件的几种方式及测试
- Java IO读写大文件的几种方式及测试
- Java IO读写大文件的几种方式及测试
- Java IO读写大文件的几种方式及测试 (2012-04-21 22:58:06)
- Java IO读写大文件的几种模式及测试
- Java读写文件的几种方式
- 【Java基础知识】IO流--字节流读写数据以及复制文件的几种方式
- 【Java基础知识】IO类--字符流读写数据以及复制文件的几种方式
- 关于SandBox机制及文件读写的几种方式
- java 文件同步读写的几种方式
- Python 读写文件的几种方式
- 黑马程序员_Java基础Day13_String类(Done)
- IQ正交调制及星座图
- preempt_disable
- ListView动态添加数据
- MYSQL删除数据库报 ERROR 1010 (HY000): Error dropping database (can't rmdir ...
- Java IO读写大文件的几种方式及测试
- socket编程中的超时设置示例详解之三(Perl实现)
- 计算机考研专业基础知识视频教程链接
- 排序算法温习 - 选择排序法
- Android UI开发第一篇——android的九宫格式实现
- HDU 4565 - 2013年长沙邀请赛 A-so easy
- servlet对象通过json返回到前台页面并展示
- 汇编小计1
- 二分图匹配算法总结