Java 拷贝文件的方式很多,除了 FileChannel 提供的方法外,还包括使用 Files.copy() 或使用字节数组的缓冲/非缓冲流。那个才是最好的选择呢?这个问题很难回答,因为答案基于很多因素。本文将目光集中到一个因素,那就是速度,因为拷贝任务 越快将会提高效率,在有些情况下,这是成功的关键。因此,本文将使用一个应用程序来比较下面这些拷贝方式的具体时间:
应用程序基于下面的条件:
下面将列出完整的应用程序:
package com.my.download;
import java.io.BufferedInputStream;
import java.io.BufferedOutputStream;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.io.RandomAccessFile;
import java.nio.ByteBuffer;
import java.nio.channels.FileChannel;
import java.nio.channels.FileChannel.MapMode;
import java.nio.file.Files;
import java.nio.file.Path;
import java.nio.file.Paths;
import java.nio.file.StandardOpenOption;
import java.util.EnumSet;
import static java.nio.file.LinkOption.NOFOLLOW_LINKS;
public class FileCopy {
private final static Path copy_from = Paths.get("C:/rafaelnadal/tournaments/2009/videos/Rafa Best Shots.mp4");
private final static Path copy_to = Paths.get("C:/Rafa Best Shots.mp4");
private static long startTime, elapsedTime;
private static int bufferSizeKB = 4;//also tested for 16, 32, 64, 128, 256 and 1024
private static int bufferSize = bufferSizeKB * 1024;
public static void main(String[] args) throws Exception {
transferfrom();
transferTo();
nonDirectBuffer();
directBuffer();
mapperedBuffer();
ioBufferedStream();
ioUnBufferedStream();
copyPath2Path();
copyInputStream2Path();
copyPath2OutputStream();
//randomReadFile();
}
public static void transferfrom() {
try (FileChannel fileChannel_from = (FileChannel.open(copy_from,
EnumSet.of(StandardOpenOption.READ)));
FileChannel fileChannel_to = (FileChannel.open(copy_to,
EnumSet.of(StandardOpenOption.CREATE_NEW, StandardOpenOption.WRITE)))) {
startTime = System.nanoTime();
fileChannel_to.transferFrom(fileChannel_from, 0L, (int) fileChannel_from.size());
elapsedTime = System.nanoTime() - startTime;
System.out.println("Elapsed Time is " + (elapsedTime / 1000000000.0) + " seconds");
}catch (IOException ex) {
System.err.println(ex);
}
deleteCopied(copy_to);
}
public static void transferTo() throws Exception{
try (FileChannel fileChannel_from = (FileChannel.open(copy_from,
EnumSet.of(StandardOpenOption.READ)));
FileChannel fileChannel_to = (FileChannel.open(copy_to,
EnumSet.of(StandardOpenOption.CREATE_NEW, StandardOpenOption.WRITE)))) {
startTime = System.nanoTime();
fileChannel_from.transferTo(0L, fileChannel_from.size(), fileChannel_to);
elapsedTime = System.nanoTime() - startTime;
System.out.println("Elapsed Time is " + (elapsedTime / 1000000000.0) + " seconds");
}catch (IOException ex) {
System.err.println(ex);
}
deleteCopied(copy_to);
}
public static void nonDirectBuffer(){
try (
FileChannel fileChannel_from = FileChannel.open(copy_from,
EnumSet.of(StandardOpenOption.READ));
FileChannel fileChannel_to = FileChannel.open(copy_to,
EnumSet.of(StandardOpenOption.CREATE_NEW, StandardOpenOption.WRITE));){
startTime = System.nanoTime();
ByteBuffer bytebuffer = ByteBuffer.allocate(bufferSize);
int bytesCount;
while ((bytesCount = fileChannel_from.read(bytebuffer)) > 0) {
bytebuffer.flip();
fileChannel_to.write(bytebuffer);
bytebuffer.clear();
}
elapsedTime = System.nanoTime() - startTime;
System.out.println("Elapsed Time is " + (elapsedTime / 1000000000.0) + " seconds");
}catch (IOException ex) {
System.err.println(ex);
}
deleteCopied(copy_to);
}
public static void directBuffer(){
try (
FileChannel fileChannel_to = FileChannel.open(copy_to,
EnumSet.of(StandardOpenOption.CREATE_NEW, StandardOpenOption.WRITE));
FileChannel fileChannel_from = (FileChannel.open(copy_from,
EnumSet.of(StandardOpenOption.READ)));) {
startTime = System.nanoTime();
ByteBuffer bytebuffer = ByteBuffer.allocateDirect(bufferSize);
int bytesCount;
while ((bytesCount = fileChannel_from.read(bytebuffer)) > 0) {
bytebuffer.flip();
fileChannel_to.write(bytebuffer);
bytebuffer.clear();
}
elapsedTime = System.nanoTime() - startTime;
System.out.println("Elapsed Time is " + (elapsedTime / 1000000000.0) + " seconds");
}catch (IOException ex) {
System.err.println(ex);
}
deleteCopied(copy_to);
}
public static void mapperedBuffer() throws Exception{
try (FileChannel fileChannel_from = (FileChannel.open(copy_from,
EnumSet.of(StandardOpenOption.READ)));
FileChannel fileChannel_to = (FileChannel.open(copy_to,
EnumSet.of(StandardOpenOption.CREATE_NEW, StandardOpenOption.WRITE)))) {
startTime = System.nanoTime();
int i=0;
long size=fileChannel_from.size()/30;
ByteBuffer rr,ww=null;
while((i<fileChannel_from.size()) && ((fileChannel_from.size()-i))>size){
rr=fileChannel_from.map(MapMode.READ_ONLY, i, size);
ww=fileChannel_to.map(MapMode.READ_WRITE, i, size);
ww.put(rr);
rr.clear();
ww.clear();
i+=size;
}
rr=fileChannel_from.map(MapMode.READ_ONLY, i, fileChannel_from.size()-i);
ww=fileChannel_to.map(MapMode.READ_WRITE, i, fileChannel_from.size()-i);
ww.put(rr);
rr.clear();
ww.clear();
elapsedTime = System.nanoTime() - startTime;
System.out.println("Elapsed Time is " + (elapsedTime / 1000000000.0) + " seconds");
}catch (IOException ex) {
System.err.println(ex);
}
deleteCopied(copy_to);
}
public static void ioBufferedStream(){
File inFileStr = copy_from.toFile();
File outFileStr = copy_to.toFile();
try (BufferedInputStream in = new BufferedInputStream(new FileInputStream(inFileStr));
BufferedOutputStream out = new BufferedOutputStream(new FileOutputStream(outFileStr))) {
startTime = System.nanoTime();
byte[] byteArray = new byte[bufferSize];
int bytesCount;
while ((bytesCount = in.read(byteArray)) != -1) {
out.write(byteArray,0, bytesCount);
}
elapsedTime = System.nanoTime() - startTime;
System.out.println("Elapsed Time is " + (elapsedTime / 1000000000.0) + " seconds");
}catch (IOException ex) {
System.err.println(ex);
}
deleteCopied(copy_to);
}
public static void ioUnBufferedStream(){
File inFileStr = copy_from.toFile();
File outFileStr = copy_to.toFile();
try (FileInputStream in = new FileInputStream(inFileStr);
FileOutputStream out = new FileOutputStream(outFileStr)) {
startTime = System.nanoTime();
byte[] byteArray = new byte[bufferSize];
int bytesCount;
while ((bytesCount = in.read(byteArray)) != -1) {
out.write(byteArray,0, bytesCount);
}
elapsedTime = System.nanoTime() - startTime;
System.out.println("Elapsed Time is " + (elapsedTime / 1000000000.0) + " seconds");
}catch (IOException ex) {
System.err.println(ex);
}
deleteCopied(copy_to);
}
public static void copyPath2Path(){
try {
startTime = System.nanoTime();
Files.copy(copy_from, copy_to, NOFOLLOW_LINKS);
elapsedTime = System.nanoTime() - startTime;
System.out.println("Elapsed Time is " + (elapsedTime / 1000000000.0) + " seconds");
}catch (IOException e) {
System.err.println(e);
}
deleteCopied(copy_to);
}
public static void copyInputStream2Path(){
try (InputStream is = new FileInputStream(copy_from.toFile())) {
startTime = System.nanoTime();
Files.copy(is, copy_to);
elapsedTime = System.nanoTime() - startTime;
System.out.println("Elapsed Time is " + (elapsedTime / 1000000000.0) + " seconds");
}catch (IOException e) {
System.err.println(e);
}
deleteCopied(copy_to);
}
public static void copyPath2OutputStream(){
try (OutputStream os = new FileOutputStream(copy_to.toFile())) {
startTime = System.nanoTime();
Files.copy(copy_from, os);
elapsedTime = System.nanoTime() - startTime;
System.out.println("Elapsed Time is " + (elapsedTime / 1000000000.0) + " seconds");
}catch (IOException e) {
System.err.println(e);
}
}
public static void randomReadFile(){
try(
RandomAccessFile read = new RandomAccessFile("C:\\Users\\asus\\Desktop\\cn_windows_7_ultimate_with_sp1_x86_dvd_618763.iso","r");
RandomAccessFile writer = new RandomAccessFile("C:\\Users\\asus\\Desktop\\dwTest\\cn_windows_7_ultimate_with_sp1_x86_dvd_618763.iso","rw");){
startTime = System.nanoTime();
byte[] b = new byte[200*1024*1024];
while(read.read(b)!=-1){
writer.write(b);
}
elapsedTime = System.nanoTime() - startTime;
System.out.println("Elapsed Time is " + (elapsedTime / 1000000000.0) + " seconds");
}catch(Exception e){
System.err.println(e);
}
deleteCopied(copy_to);
}
public static void deleteCopied(Path path){
try {
Files.deleteIfExists(path);
}catch (IOException ex) {
System.err.println(ex);
}
}
}
输出结果排序比较复杂,其中包含了很多数据。下面我将主要的对比用图形的方式展示出来。图形中 Y 坐标表示消耗的时间(单位:秒),X 坐标表示缓冲的大小(或运行次数,跳过了前三次运行)。
从下图看来,如果缓存小于 256KB,那么非直接模式的 Buffer 快一点,而缓存大于 256KB 后,直接模式的 Buffer 快一点:
从下图看来,FileChannel.transferTo() 和 FileChannel.transferFrom 运行七次的速度都差不多,而 FileChannel.map 的速度就要差很多:
从下图看来,最快的是 Path 到 Path,其次是 Path 到 OutputStream,最慢的是 InputStream 到 Path:
最后,我们将前面最快的三种方式综合起来比较。从比较的结果来看,似乎 Path 到 Path 是最快的解决方案: