java多線程下載文件原理解析
更新時間:2019年04月18日 11:41:23 作者:ITzhongzi
這篇文章主要為大家詳細介紹了java多線程下載文件原理,具有一定的參考價值,感興趣的小伙伴們可以參考一下
原理解析:利用RandomAccessFile在本地創(chuàng)建一個隨機訪問文件,文件大小和服務器要下載的文件大小相同。根據(jù)線程的數(shù)量(假設有三個線程),服務器的文件三等分,并把我們在本地創(chuàng)建的文件同樣三等分,每個線程下載自己負責的部分,到相應的位置即可。
示例圖:

示例demo
import java.io.InputStream;
import java.io.RandomAccessFile;
import java.net.HttpURLConnection;
import java.net.URL;
public class MutilDownload {
private static String path = "http://192.168.80.85:8080/test.doc";
private static final int threadCount = 3;
public static void main(String[] args) {
try {
URL url = new URL(path);
HttpURLConnection conn = (HttpURLConnection) url.openConnection();
conn.setRequestMethod("GET");
conn.setConnectTimeout(5000);
int responseCode = conn.getResponseCode();
if (responseCode == 200) {
int contentLength = conn.getContentLength();
System.out.println("length" + contentLength);
RandomAccessFile rafAccessFile = new RandomAccessFile("test.doc", "rw");
rafAccessFile.setLength(contentLength);
int blockSize = contentLength / threadCount;
for (int i = 0; i < threadCount; i++) {
int startIndex = i * blockSize; //每個現(xiàn)成下載的開始位置
int endIndex = (i + 1) * blockSize - 1;// 每個線程的結束位置
if (i == threadCount - 1) {
//最后一個線程
endIndex = contentLength - 1;
}
new DownloadThread(startIndex, endIndex, i).start();
}
}
} catch (Exception e) {
}
}
private static class DownloadThread extends Thread {
private int startIndex;
private int endIndex;
private int threadId;
public DownloadThread(int startIndex, int endIndex, int threadId) {
this.startIndex = startIndex;
this.endIndex = endIndex;
this.threadId = threadId;
}
@Override
public void run() {
try {
URL url = new URL(path);
HttpURLConnection conn = (HttpURLConnection) url.openConnection();
conn.setRequestMethod("GET");
conn.setConnectTimeout(5000);
conn.setRequestProperty("Range", "bytes=" + startIndex + "-" + endIndex); //固定寫法,請求部分資源
int responseCode = conn.getResponseCode(); // 206表示請求部分資源
if (responseCode == 206) {
RandomAccessFile rafAccessFile = new RandomAccessFile("test.doc", "rw");
rafAccessFile.seek(startIndex);
InputStream is = conn.getInputStream();
int len = -1;
byte[] buffer = new byte[1024];
while ((len = is.read(buffer)) != -1) {
rafAccessFile.write(buffer, 0, len);
}
rafAccessFile.close();
System.out.println("線程" + threadId + "下載完成");
}
} catch (Exception e) {
}
}
}
}
以上就是本文的全部內(nèi)容,希望對大家的學習有所幫助,也希望大家多多支持腳本之家。
相關文章
Spring Boot集成ShedLock分布式定時任務的實現(xiàn)示例
ShedLock確保您計劃的任務最多同時執(zhí)行一次。如果一個任務正在一個節(jié)點上執(zhí)行,則它會獲得一個鎖,該鎖將阻止從另一個節(jié)點(或線程)執(zhí)行同一任務。2021-05-05
Java數(shù)據(jù)結構和算法之冒泡,選擇和插入排序算法
這篇文章主要為大家介紹了Java冒泡,選擇和插入排序算法 ,具有一定的參考價值,感興趣的小伙伴們可以參考一下,希望能夠給你帶來幫助2022-01-01
Java性能優(yōu)化之數(shù)據(jù)結構實例代碼
這篇文章主要介紹了Java性能優(yōu)化之數(shù)據(jù)結構實例代碼,具有一定借鑒價值,需要的朋友可以參考下2018-01-01

