读取文件大小:1.45G
第一种,OldIO:
第一种,OldIO:
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 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();
}
相对于最后一种内存直接映射方式前面的测试其实无意义,基本秒杀。。。。。
对于很大的文件直接分块映射时内存会不够,这是因为MappedByteBuffer未被释放造成的,sun未提供直接回收MappedByteBuffer区域的方法,这个时候有两种方法解决,第一种比较愚笨的:System.gc(); System.runFinalization(); try { Thread.sleep(3000); } catch (InterruptedException e) { e.printStackTrace(); }
- 第二种网上找来的,利用反射调用clean方法:
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; } }); }