真实的国产乱ⅩXXX66竹夫人,五月香六月婷婷激情综合,亚洲日本VA一区二区三区,亚洲精品一区二区三区麻豆

成都創(chuàng)新互聯(lián)網(wǎng)站制作重慶分公司

java中文件拷貝流的介紹-創(chuàng)新互聯(lián)

最近項(xiàng)目里有個(gè)需求需要實(shí)現(xiàn)文件拷貝,在java中文件拷貝流的讀寫,很容易就想到IO中的InputStream和OutputStream之類的,但是上網(wǎng)查了一下文件拷貝也是有很多種方法的,除了IO,還有NIO、Apache提供的工具類、JDK自帶的文件拷貝方法

創(chuàng)新互聯(lián)公司是一家以網(wǎng)絡(luò)技術(shù)公司,為中小企業(yè)提供網(wǎng)站維護(hù)、網(wǎng)站設(shè)計(jì)制作、成都網(wǎng)站建設(shè)、網(wǎng)站備案、服務(wù)器租用、域名注冊(cè)、軟件開發(fā)、微信小程序定制開發(fā)等企業(yè)互聯(lián)網(wǎng)相關(guān)業(yè)務(wù),是一家有著豐富的互聯(lián)網(wǎng)運(yùn)營(yíng)推廣經(jīng)驗(yàn)的科技公司,有著多年的網(wǎng)站建站經(jīng)驗(yàn),致力于幫助中小企業(yè)在互聯(lián)網(wǎng)讓打出自已的品牌和口碑,讓企業(yè)在互聯(lián)網(wǎng)上打開一個(gè)面向全國(guó)乃至全球的業(yè)務(wù)窗口:建站歡迎聯(lián)系:028-86922220

IO拷貝

public class IOFileCopy {

   private static final int BUFFER_SIZE = 1024;

   public static void copyFile(String source, String target) {
     long start = System.currentTimeMillis();
     try(InputStream in = new FileInputStream(new File(source));
       OutputStream out = new FileOutputStream(new File(target))) {
       byte[] buffer = new byte[BUFFER_SIZE];
       int len;
       while ((len = in.read(buffer)) > 0) {
         out.write(buffer, 0, len);
       }

       System.out.println(String.format("IO file copy cost %d msc", System.currentTimeMillis() - start));
     } catch (Exception e) {
       e.printStackTrace();
     }
   }
}

傳統(tǒng)IO中文件讀取過程可以分為以下幾步:

  • 內(nèi)核從磁盤讀取數(shù)據(jù)到緩沖區(qū),這個(gè)過程由磁盤操作器通過DMA操作將數(shù)據(jù)從磁盤讀取到內(nèi)核緩沖區(qū),該過程不依賴CPU

  • 用戶進(jìn)程在將數(shù)據(jù)從內(nèi)核緩沖區(qū)拷貝到用戶空間緩沖區(qū)

  • 用戶進(jìn)程從用戶空間緩沖區(qū)讀取數(shù)據(jù)

java中文件拷貝流的介紹

NIO拷貝

NIO進(jìn)行文件拷貝有兩種實(shí)現(xiàn)方式,一是通過管道,而是通過文件內(nèi)存內(nèi)存映射

public class NIOFileCopy {

   public static void copyFile(String source, String target) {
     long start = System.currentTimeMillis();
     try(FileChannel input = new FileInputStream(new File(source)).getChannel();
       FileChannel output = new FileOutputStream(new File(target)).getChannel()) {
       output.transferFrom(input, 0, input.size());
     } catch (Exception e) {
       e.printStackTrace();
     }

     System.out.println(String.format("NIO file copy cost %d msc", System.currentTimeMillis() - start));
   }
}

文件內(nèi)存映射:

把內(nèi)核空間地址與用戶空間的虛擬地址映射到同一個(gè)物理地址,DMA 硬件可以填充對(duì)內(nèi)核與用戶空間進(jìn)程同時(shí)可見的緩沖區(qū)了。用戶進(jìn)程直接從內(nèi)存中讀取文件內(nèi)容,應(yīng)用只需要和內(nèi)存打交道,不需要進(jìn)行緩沖區(qū)來(lái)回拷貝,大大提高了IO拷貝的效率。加載內(nèi)存映射文件所使用的內(nèi)存在Java堆區(qū)之外

public class NIOFileCopy2 {

   public static void copyFile(String source, String target) {
     long start = System.currentTimeMillis();
     try(FileInputStream fis = new FileInputStream(new File(source));
       FileOutputStream fos = new FileOutputStream(new File(target))) {
       FileChannel sourceChannel = fis.getChannel();
       FileChannel targetChannel = fos.getChannel();
       MappedByteBuffer mappedByteBuffer = sourceChannel.map(FileChannel.MapMode.READ_ONLY, 0, sourceChannel.size());
       targetChannel.write(mappedByteBuffer);
     } catch (FileNotFoundException e) {
       e.printStackTrace();
     } catch (IOException e) {
       e.printStackTrace();
     }

     System.out.println(String.format("NIO memory reflect file copy cost %d msc", System.currentTimeMillis() - start));
     File targetFile = new File(target);
     targetFile.delete();
   }
}

NIO內(nèi)存映射文件拷貝可以分為以下幾步

java中文件拷貝流的介紹

NIO的內(nèi)存映射實(shí)際上就是少了一次從內(nèi)核空間拷貝到用戶空間的過程,將對(duì)用戶緩沖區(qū)的讀改為從內(nèi)存讀取

Files#copyFile方法

public class FilesCopy {

   public static void copyFile(String source, String target) {
     long start = System.currentTimeMillis();
     try {
       File sourceFile = new File(source);
       File targetFile = new File(target);
       Files.copy(sourceFile.toPath(), targetFile.toPath());
     } catch (IOException e) {
       e.printStackTrace();
     }

     System.out.println(String.format("FileCopy file copy cost %d msc", System.currentTimeMillis() - start));
   }
}

FileUtils#copyFile方法

使用FileUtils之前需先引入依賴

  • 依賴

     
       commons-io
       commons-io
       2.4
    
  • FileUtils#copyFile封裝類:FileUtilsCopy.java

    public class FileUtilsCopy {
    
       public static void copyFile(String source, String target) {
         long start = System.currentTimeMillis();
         try {
           FileUtils.copyFile(new File(source), new File(target));
         } catch (IOException e) {
           e.printStackTrace();
         }
    
         System.out.println(String.format("FileUtils file copy cost %d msc", System.currentTimeMillis() - start));
       }
    }

性能比較

既然有這么多種實(shí)現(xiàn)方法,肯定要從中選擇性能最佳的

測(cè)試環(huán)境:

  • windows 10
  • CPU 6核
  • JDK1.8

測(cè)試代碼:PerformTest.java

public class PerformTest {

   private static final String source1 = "input/test1.txt";
   private static final String source2 = "input/test2.txt";
   private static final String source3 = "input/test3.txt";
   private static final String source4 = "input/test4.txt";
   private static final String target1 = "output/test1.txt";
   private static final String target2 = "output/test2.txt";
   private static final String target3 = "output/test3.txt";
   private static final String target4 = "output/test4.txt";

   public static void main(String[] args) {
     IOFileCopy.copyFile(source1, target1);
     NIOFileCopy.copyFile(source2, target2);
     FilesCopy.copyFile(source3, target3);
     FileUtilsCopy.copyFile(source4, target4);
   }
}

總共執(zhí)行了五次,讀寫的文件大小分別為9KB、23KB、239KB、1.77MB、12.7MB

java中文件拷貝流的介紹

注意:?jiǎn)挝痪鶠楹撩?/p>

從執(zhí)行結(jié)果來(lái)看:

  • 文件很小時(shí) => IO > NIO【內(nèi)存映射】> NIO【管道】 > Files#copy > FileUtils#copyFile

  • 在文件較小時(shí) => NIO【內(nèi)存映射】> IO > NIO【管道】 > Files#copy > FileUtils#copyFile

  • 在文件較大時(shí) => NIO【內(nèi)存映射】> > NIO【管道】> IO > Files#copy > FileUtils#copyFile

  • 修改IO緩沖區(qū)大小對(duì)拷貝效率有影響,但是并不是越大性能越好,稍大于拷貝文件大小即可

文件較小時(shí),IO效率高于NIO,NIO底層實(shí)現(xiàn)較為復(fù)雜,NIO的優(yōu)勢(shì)不明顯。同時(shí)NIO內(nèi)存映射初始化耗時(shí),所以在文件較小時(shí)和IO復(fù)制相比沒有優(yōu)勢(shì)

如果追求效率可以選擇NIO的內(nèi)存映射去實(shí)現(xiàn)文件拷貝,但是對(duì)于大文件使用內(nèi)存映射拷貝要格外關(guān)注系統(tǒng)內(nèi)存的使用率。推薦:大文件拷貝使用內(nèi)存映射,原文是這樣的:

For most operating systems, mapping a file into memory is more
expensive than reading or writing a few tens of kilobytes of data via
the usual {@link #read read} and {@link #write write} methods.  From the
standpoint of performance it is generally only worth mapping relatively
large files into memory

絕大多數(shù)操作系統(tǒng)的內(nèi)存映射開銷大于IO開銷

另外有需要云服務(wù)器可以了解下創(chuàng)新互聯(lián)scvps.cn,海內(nèi)外云服務(wù)器15元起步,三天無(wú)理由+7*72小時(shí)售后在線,公司持有idc許可證,提供“云服務(wù)器、裸金屬服務(wù)器、高防服務(wù)器、香港服務(wù)器、美國(guó)服務(wù)器、虛擬主機(jī)、免備案服務(wù)器”等云主機(jī)租用服務(wù)以及企業(yè)上云的綜合解決方案,具有“安全穩(wěn)定、簡(jiǎn)單易用、服務(wù)可用性高、性價(jià)比高”等特點(diǎn)與優(yōu)勢(shì),專為企業(yè)上云打造定制,能夠滿足用戶豐富、多元化的應(yīng)用場(chǎng)景需求。


分享名稱:java中文件拷貝流的介紹-創(chuàng)新互聯(lián)
轉(zhuǎn)載注明:http://weahome.cn/article/ehpeh.html

其他資訊

在線咨詢

微信咨詢

電話咨詢

028-86922220(工作日)

18980820575(7×24)

提交需求

返回頂部