亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

Java利用多線程復制文件

 更新時間:2022年09月16日 09:46:55   作者:發(fā)光吖  
這篇文章主要介紹了Java利用多線程復制文件,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下

前言

復制一個文件,是學習IO流時最基本的操作。你可以使用字節(jié)型文件流,也可以使用高級緩沖流。

但是,它們都是單線程的。

如果需要復制一個大型文件,單線程的復制一般而言是不能夠充分發(fā)揮CPU以及內(nèi)存的性能。這時候就需要利用多線程來復制文件。

多線程的讀:

我們很自然地想到,利用FileInputStream類的skip()方法,可以跳著讀,這就對多線程比較友好,啟動多個線程,第一個線程讀一部分,第二個線程跳過一部分字節(jié)再讀,這沒有問題。

但是如果要寫呢?我們知道FileOutputStream類是沒有與skip類似的方法的,也就是說,它不能跳著寫,這就很麻煩。

這就意味著,如果需要利用多線程復制一個文件,那么首先得把這個文件利用多線程并發(fā),讀取并同時寫入成多個文件碎片,然后再利用單線程去一一讀取這些文件碎片,把它們的內(nèi)容寫入到一個完整的文件中。必要的話,最后再刪除這些中間文件碎片。

這個過程,復雜,且性能不高。涉及到兩套讀寫,前面多線程讀寫,后面單線程讀寫。很顯然這個過程雖然使用到了多線程但它不能提高性能,反而降低了性能。

既然雞肋點在于FileOutputStream不能跳著寫,那么就找一個能跳著寫的類吧。

RandomAccessFile

RandomAccessFile是java Io體系中功能最豐富的文件內(nèi)容訪問類。即可以讀取文件內(nèi)容,也可以向文件中寫入內(nèi)容。但是和其他輸入/輸入流不同的是,程序可以直接跳到文件的任意位置來讀寫數(shù)據(jù)。

RandomAccessFile包含了以下兩個方法來操作文件的記錄指針:

  • long getFilePointer(); 返回文件記錄指針的當前位置
  • void seek(long pos); 將文件記錄指針定位到pos位置

有了RandomAccessFile這個類,上面的問題就迎刃而解,因為它最大的好處就是可以實現(xiàn)從指定位置寫入一些數(shù)據(jù)到文件中!

順便說一下它的構造方法:

  • RandomAccessFile(File file, String mode)
  • RandomAccessFile(String name, String mode)

mode表示RandomAccessFile的訪問模式,她有4個值:

  • “r”:只有讀的權限,如果你試圖去執(zhí)行寫入方法,則拋出IOException異常。
  • “rw”:有讀和寫 兩個權限,如果該文件不存在,則會創(chuàng)建該文件。
  • “rws”:相對于”rw” 模式,還要求對文件內(nèi)容或元數(shù)據(jù)的每個更新都同步寫入到底層設備。
  • “rwd”:相對于”rw” 模式,還要求對文件內(nèi)容每個更新都同步寫入到底層設備。

一般而言我們使用“r”和“rw”就夠了,后面那兩個我也不懂干嘛的。

代碼

package testThread.file_threading;

import java.io.BufferedInputStream;
import java.io.FileInputStream;
import java.io.IOException;
import java.io.RandomAccessFile;

public class ThreadFileCopy extends Thread {
? ? private String srcFileStr;//源文件的路徑
? ? private String desFileStr;//目標文件的路徑 ,des --> destination目的地
? ? private long skipLen;//跳過多少個字節(jié)開始讀/寫
? ? private long workload;//總共要讀/寫多少個字節(jié)
? ? private final int IO_UNIT = 1024;//每次讀寫的基本單位(1024個字節(jié))

? ? public ThreadFileCopy(String srcFileStr, String desFileStr, long skipLen, long workload) {
? ? ? ? this.srcFileStr = srcFileStr;
? ? ? ? this.desFileStr = desFileStr;
? ? ? ? this.skipLen = skipLen;
? ? ? ? this.workload = workload;
? ? }
? ? public void run(){
? ? ? ? FileInputStream fis = null;
? ? ? ? BufferedInputStream bis = null;//利用高級緩沖流,加快讀的速度
? ? ? ? RandomAccessFile raf = null;
? ? ? ? try {
? ? ? ? ? ? fis = new FileInputStream(srcFileStr);
? ? ? ? ? ? bis = new BufferedInputStream(fis);
? ? ? ? ? ? raf = new RandomAccessFile(desFileStr,"rw");
? ? ? ? ? ? bis.skip(this.skipLen);//跳過一部分字節(jié)開始讀
? ? ? ? ? ? raf.seek(this.skipLen);//跳過一部分字節(jié)開始寫
? ? ? ? ? ? byte[] bytes = new byte[IO_UNIT];
? ? ? ? ? ? //根據(jù)總共需要復制的字節(jié)數(shù) 和 讀寫的基本單元 計算出一共需要讀寫的次數(shù),利用讀寫次數(shù)控制循環(huán)
? ? ? ? ? ? long io_num = this.workload/IO_UNIT + 1;//因為workload/1024 很可能不能整除,會有余數(shù)
? ? ? ? ? ? if(this.workload % IO_UNIT == 0)
? ? ? ? ? ? ? ? io_num--;//如果碰巧整除,讀寫次數(shù)減一
? ? ? ? ? ? //count表示讀取的有效字節(jié)數(shù),雖然count不參與控制循環(huán)結(jié)束,
? ? ? ? ? ? // 但是它能有效避免最后一次讀取出的byte數(shù)組中有大量空字節(jié)寫入到文件中,導致復制出的文件稍稍變大
? ? ? ? ? ? int count = bis.read(bytes);
? ? ? ? ? ? while (io_num != 0){
? ? ? ? ? ? ? ? raf.write(bytes,0,count);
? ? ? ? ? ? ? ? count = bis.read(bytes,0,count);//重新計算count的值
? ? ? ? ? ? ? ? io_num--;
? ? ? ? ? ? }
? ? ? ? } catch (IOException e) {
? ? ? ? ? ? e.printStackTrace();
? ? ? ? }finally {
? ? ? ? ? ? try {
? ? ? ? ? ? ? ? if (fis != null)
? ? ? ? ? ? ? ? ? ? fis.close();
? ? ? ? ? ? } catch (IOException e) {
? ? ? ? ? ? ? ? e.printStackTrace();
? ? ? ? ? ? }
? ? ? ? ? ? try {
? ? ? ? ? ? ? ? if (bis != null)
? ? ? ? ? ? ? ? ? ? bis.close();
? ? ? ? ? ? } catch (IOException e) {
? ? ? ? ? ? ? ? e.printStackTrace();
? ? ? ? ? ? }
? ? ? ? ? ? try {
? ? ? ? ? ? ? ? if (raf != null)
? ? ? ? ? ? ? ? ? ? raf.close();
? ? ? ? ? ? } catch (IOException e) {
? ? ? ? ? ? ? ? e.printStackTrace();
? ? ? ? ? ? }
? ? ? ? }
? ? }
}

測試

package testThread.file_threading;

import java.io.File;

public class TestMain {
? ? public static void main(String[] args) {
? ? ? ? int thread_num = 5;//創(chuàng)建5個線程讀寫
? ? ? ? String srcFileStr = "E:\\test\\123.flv";
? ? ? ? String desFileStr = "E:\\test\\thread.flv";
? ? ? ? File srcFile = new File(srcFileStr);
? ? ? ? long workload = srcFile.length()/thread_num;//總共要讀/寫多少個字節(jié)
? ? ? ? //用一個數(shù)組來存儲每個線程跳過的字節(jié)數(shù)
? ? ? ? long[] skipLenArr = new long[thread_num];
? ? ? ? for(int i = 0;i<skipLenArr.length;i++){
? ? ? ? ? ? skipLenArr[i] = i*workload;
? ? ? ? }
? ? ? ? //用一個數(shù)組來存儲所有的線程
? ? ? ? ThreadFileCopy[] tfcs = new ThreadFileCopy[thread_num];
? ? ? ? //初始化所有線程
? ? ? ? for(int i = 0;i<tfcs.length;i++){
? ? ? ? ? ? tfcs[i] = new ThreadFileCopy(srcFileStr,desFileStr,skipLenArr[i],workload);
? ? ? ? }
? ? ? ? //讓所有線程進入就緒狀態(tài)
? ? ? ? for(int i = 0;i<tfcs.length;i++){
? ? ? ? ? ? tfcs[i].start();
? ? ? ? }
? ? ? ? System.out.println("復制完畢!");
? ? }
}

以上就是本文的全部內(nèi)容,希望對大家的學習有所幫助,也希望大家多多支持腳本之家。

相關文章

  • java String 可變性的分析

    java String 可變性的分析

    這篇文章主要介紹了java String 可變性的分析的相關資料,通常大家都認為java String 是不可變的,這里分析下源碼來說明它的可變性,需要的朋友可以參考下
    2017-03-03
  • Java中過濾器、監(jiān)聽器和攔截器的區(qū)別詳解

    Java中過濾器、監(jiān)聽器和攔截器的區(qū)別詳解

    這篇文章主要介紹了Java中過濾器、監(jiān)聽器和攔截器的區(qū)別詳解,有些朋友可能不了解過濾器、監(jiān)聽器和攔截器的區(qū)別,本文就來詳細講一下,相信看完你會有所收獲,需要的朋友可以參考下
    2024-01-01
  • java 中接口和抽象類的區(qū)別與對比

    java 中接口和抽象類的區(qū)別與對比

    這篇文章主要介紹了java 中接口和抽象類的區(qū)別與對比的相關資料,這里詳細說明他們之家的區(qū)別,需要的朋友可以參考下
    2017-08-08
  • Spring事件監(jiān)聽器ApplicationListener源碼詳解

    Spring事件監(jiān)聽器ApplicationListener源碼詳解

    這篇文章主要介紹了Spring事件監(jiān)聽器ApplicationListener源碼詳解,ApplicationEvent以及Listener是Spring為我們提供的一個事件監(jiān)聽、訂閱的實現(xiàn),內(nèi)部實現(xiàn)原理是觀察者設計模式,需要的朋友可以參考下
    2023-05-05
  • Java中ByteArrayOutputStream亂碼問題解決

    Java中ByteArrayOutputStream亂碼問題解決

    本文主要介紹了Java中ByteArrayOutputStream亂碼問題解決,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2023-07-07
  • 淺析java中stringBuilder的用法

    淺析java中stringBuilder的用法

    下面小編就為大家?guī)硪黄獪\析java中stringBuilder的用法。小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考,一起跟隨小編過來看看吧
    2016-05-05
  • 基于SpringBoot實現(xiàn)自動裝配返回屬性的設計思路

    基于SpringBoot實現(xiàn)自動裝配返回屬性的設計思路

    這篇文章主要介紹了基于SpringBoot實現(xiàn)自動裝配返回屬性,這里涉及到的技術知識點有注解解析器,為什么用ResponseBodyAdvice這里解析?不在Filter,Interceptors,本文結(jié)合示例代碼給大家介紹的非常詳細,需要的朋友參考下吧
    2022-03-03
  • 深入了解Java Object類的使用

    深入了解Java Object類的使用

    java繼承中說到的Object類是java中一個特殊的類,所有的類都是直接或者間接的繼承自Object類。本文就和大家詳細講講Java Object類的使用,感興趣的可以了解一下
    2022-07-07
  • Spring中@RequestMapping、@PostMapping、@GetMapping的實現(xiàn)

    Spring中@RequestMapping、@PostMapping、@GetMapping的實現(xiàn)

    RequestMapping、@PostMapping和@GetMapping是三個非常常用的注解,本文就來介紹一下這三種注解的具體使用,具有一定的參考價值,感興趣的可以了解一下
    2024-07-07
  • Java中finalize()詳解及用法

    Java中finalize()詳解及用法

    這篇文章主要介紹了Java中finalize()詳解及用法的相關資料,final是Java的關鍵字,它所表示的是“這部分是無法修改的”,需要的朋友可以參考下
    2017-03-03

最新評論