java实现一个多线程下载工具类

本文介绍了一种基于Java的多线程下载技术实现方案。利用URL对象的openStream()方法读取远程资源,并通过创建多个线程并行下载来提高下载速度。文中详细解释了如何划分文件块、设置HTTP请求头以及进度监控。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

URL对象中的前面几个方法都非常容易理解,而该对象提供的openStream()方法可以就读该URL资源的InputStream,通过该方法可以非常方便地读取远程资源--甚至实现多线程下载。


import java.io.InputStream;
import java.io.RandomAccessFile;
import java.net.HttpURLConnection;
import java.net.URL;


public class DownUtil {
//定义下载资源的路径
private String path;
//指定所下载的文件的保存位置
private String targetFile;
//定义需要使用多少个线程下载资源
private int threadNum;
//定义下载的线程对象
private DownThread[] threads;
//定义下载的文件的总大小
private int fileSize;


public DownUtil(String path,String targetFile,int threadNum){
this.path=path;
this.threadNum =threadNum;
//初始化threads数组
threads =new DownThread(threadNum);
this.targetFile=targetFile;
}

public void download() throws Exception{
URL url =new URL(path);
HttpURLConnection conn =(HttpURLConnection) url.openConnection();
conn.setConnectTimeout(5*1000);
conn.setRequestMethod("GET");
conn.setRequestProperty("Accept", 
"image/gif,image/jpeg,image/pjpeg,image/pjpeg"
+"application/x-shockwave-flish,application/xam+xml"
+"application/vnd.ms-xpsdocument,application/x-ms-xbap,"
+"application/x-ms-application,application/vnd.ms-excel,"
+"application/vnd.ms-powerpoint,application/msword,*/*");
conn.setRequestProperty("Accept-Language", "zh-CN");
conn.setRequestProperty("Charset", "UTF-8");
conn.setRequestProperty("Connection", "Keep-Alive");
//得到文件大小
fileSize = conn.getContentLength();
conn.disconnect();
int currentPartSize = fileSize / threadNum+1;
RandomAccessFile file = new RandomAccessFile(targetFile,"rw");
//设置本地文件的大小
file.setLength(fileSize);
file.close();
for (int i=0;i<threadNum;i++){
//计算每个线程下载的开始位置
int startPos = i*currentPartSize;
//每个线程使用一个RandomAccessFile进行下载
RandomAccessFile currentPart = new RandomAccessFile(targetFile,"rw");
//定位该线程的下载位置
currentPart.seek(startPos);
//创建下载线程
threads[i] =new DownThread(startPos,currentPartSize,currentPart);
//启动下载线程
threads[i].start();
}
}

//获取下载的完成百分比
public double getCompleteRate(){
//统计多个线程已经下载的总大小
int sumSize =0;
for (int i=0;i<threadNum;i++){
sumSize+=threads[i].length;
}
//返回一已经完成的百分比
return sumSize*1.0/fileSize;
}
private class DownThread extends Thread{
//当前线程的下载位置
private int startPos;
//定义当前线程负责下载的文件大小
private int currentPartSize;
//当前线程需要下载的文件块
private RandomAccessFile currentPart;
//定义该县城已下载的字节数
public int length;
public DownThread(int startPos,int currentPartSize,RandomAccessFile currentPart){
this.startPos=startPos;
this.currentPartSize=currentPartSize;
this.currentPart=currentPart;

}
public void run(){
try{
URL url =new URL(path);
HttpURLConnection conn=(HttpURLConnection)url.openConnection();
conn.setConnectTimeout(5*1000);
conn.setRequestMethod("GET");
conn.setRequestProperty("Accept", 
"image/gif,image/jpeg,image/pjpeg,image/pjpeg"
+"application/x-shockwave-flish,application/xam+xml"
+"application/vnd.ms-xpsdocument,application/x-ms-xbap,"
+"application/x-ms-application,application/vnd.ms-excel,"
+"application/vnd.ms-powerpoint,application/msword,*/*");
conn.setRequestProperty("Accept-Language", "zh-CN");
conn.setRequestProperty("Charset", "UTF-8");
InputStream inStream = conn.getInputStream();
//跳过startPos个字节,表明该线程只下载自己负责的那部分文件
inStream.skip(this.startPos);
byte[] buffer=new byte[1024];
int hasRead =0;
//读取网络数据,并写入本地文件
while(length<currentPartSize&&(hasRead = inStream.read(buffer))!=-1){
currentPart.write(buffer,0,hasRead);
//累计该线程下载的总大小
length+=hasRead;
}
currentPart.close();
inStream.close();
}
catch(Exception e){
e.printStackTrace();
}
}
}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值