Java多線程下載的實(shí)現(xiàn)方法
package cn.me.test;
import java.io.InputStream;
import java.io.RandomAccessFile;
import java.net.HttpURLConnection;
import java.net.URL;
/**
* 多線程下載
* 1:使用RandomAccessFile在任意的位置寫(xiě)入數(shù)據(jù)。
* 2:需要計(jì)算第一個(gè)線程下載的數(shù)據(jù)量,可以平均分配。如果不夠平均時(shí),
* 則直接最后一個(gè)線程處理相對(duì)較少的數(shù)據(jù)
* 3:必須要在下載之前準(zhǔn)備好相同大小的文件,通過(guò)文件頭獲取
*/
public class MultiThreadDownload {
public static void main(String[] args) throws Exception {
//1:聲明文件名和下載的地址
String fileName = "aa.rar";
String urlStr = "http://localhost:7777/day18";
//2:聲明Url
URL url = new URL(urlStr+"/"+fileName);
//3:獲取連接
HttpURLConnection con =
(HttpURLConnection) url.openConnection();
//4:設(shè)置請(qǐng)求方式
con.setRequestMethod("GET");
//5:獲取請(qǐng)求頭,即文件的長(zhǎng)度
int length = con.getContentLength();//獲取下載文件的長(zhǎng)度,以計(jì)算每個(gè)線程應(yīng)該下載的數(shù)據(jù)量。
//6:在指定的目錄下,創(chuàng)建一個(gè)同等大小的文件
RandomAccessFile file = new RandomAccessFile("d:/a/"+fileName, "rw");//創(chuàng)建一個(gè)相同大小的文件。
//7:設(shè)置文件大小,占位
file.setLength(length);//設(shè)置文件大小。
file.close();
//8:定義線程個(gè)數(shù)
int size = 3;
//9:計(jì)算每一個(gè)線程應(yīng)該下載多少字節(jié)的數(shù)據(jù),如果正好整除則最好,否則加1
int block = length/size==0?length/size:length/size+1;//計(jì)算每個(gè)線程應(yīng)該下載的數(shù)據(jù)量。
System.err.println("每個(gè)線程應(yīng)該下載:"+block);
//10:運(yùn)行三個(gè)線程并計(jì)算從哪個(gè)字節(jié)開(kāi)始到哪一個(gè)字節(jié)結(jié)束
for(int i=0;i<size;i++){
int start = i*block;
int end = start+(block-1);//計(jì)算每一個(gè)線程的開(kāi)始和結(jié)束字節(jié)。
System.err.println(i+"="+start+","+end);
new MyDownThread(fileName, start, end,url).start();
}
}
static class MyDownThread extends Thread{
//定義文件名
private String fileName;
//定義從何地開(kāi)始下載
private int start;
//定義下載到哪一個(gè)字節(jié)
private int end;
private URL url;
public MyDownThread(String fileName,int start,int end,URL url){
this.fileName=fileName;
this.start=start;
this.end=end;
this.url=url;
}
@Override
public void run() {
try{
//11:開(kāi)始下載
HttpURLConnection con =
(HttpURLConnection) url.openConnection();
con.setRequestMethod("GET");
//12:設(shè)置分段下載的請(qǐng)求頭
con.setRequestProperty("Range","bytes="+start+"-"+end);//設(shè)置從服務(wù)器上讀取的文件塊。
//13:開(kāi)始下載,需要判斷206
if(con.getResponseCode()==206){//訪問(wèn)成功,則返回的狀態(tài)碼為206。
InputStream in = con.getInputStream();
//14:聲明隨機(jī)寫(xiě)文件對(duì)象,注意rwd是指即時(shí)將數(shù)據(jù)寫(xiě)到文件中,而不使用緩存區(qū)
RandomAccessFile out = new RandomAccessFile("d:/a/"+fileName,"rwd");
out.seek(start);//設(shè)置從文件的某個(gè)位置開(kāi)始寫(xiě)數(shù)據(jù)。
byte[] b=new byte[1024];
int len = 0;
while((len=in.read(b))!=-1){
out.write(b,0,len);
}
out.close();
in.close();
}
System.err.println(this.getName()+"執(zhí)行完成");
}catch(Exception e){
throw new RuntimeException(e);
}
}
}
}
相關(guān)文章
詳細(xì)理解JAVA面向?qū)ο蟮姆庋b,繼承,多態(tài),抽象
這篇文章主要介紹了Java基礎(chǔ)之面向?qū)ο髾C(jī)制(多態(tài)、繼承)底層實(shí)現(xiàn),文中有非常詳細(xì)的代碼示例,對(duì)正在學(xué)習(xí)java的小伙伴們有非常好的幫助,需要的朋友可以參考下2021-07-07java實(shí)現(xiàn)的密碼強(qiáng)度檢測(cè)功能完整示例
這篇文章主要介紹了java實(shí)現(xiàn)的密碼強(qiáng)度檢測(cè)功能,結(jié)合完整實(shí)例形式分析了java針對(duì)密碼強(qiáng)度檢測(cè)相關(guān)的字符串遍歷、判斷,以及輸出密碼強(qiáng)度等級(jí)相關(guān)操作技巧,需要的朋友可以參考下2019-06-06MyBatis接口的簡(jiǎn)單實(shí)現(xiàn)原理分析
這里僅僅舉個(gè)簡(jiǎn)單例子來(lái)說(shuō)明原理,不是完全針對(duì)MyBatis的,這種思想我們也可以應(yīng)用在其他地方。地mybatis接口實(shí)現(xiàn)原理感興趣的朋友一起看看吧2017-07-07如何利用Stream改變list中特定對(duì)象的某一屬性
這篇文章主要介紹了如何利用Stream改變list中特定對(duì)象的某一屬性問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-12-12mybatis-plus開(kāi)啟sql日志打印的三種方法
本文主要介紹了mybatis-plus開(kāi)啟sql日志打印的三種方法,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2023-05-05常用json與javabean互轉(zhuǎn)的方法實(shí)現(xiàn)
這篇文章主要介紹了常用json與javabean互轉(zhuǎn)的方法實(shí)現(xiàn),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2021-04-04