首先,我们构建一个多线程下载工具类--DownUtil.代码如下:

import java.net.URL;
import java.net.HttpURLConnection;
import java.io.InputStream ;
import java.io.RandomAccessFile;
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);
//1.通过调用URL对象的openConnection()方法来创建URLConnection对象
HttpURLConnection conn=(HttpURLConnection)url.openConnection();
//2.设置URLConnection的参数和普通请求属性
conn.setConnectTimeout(5*1000);
conn.setRequestMethod("GET");
conn.setRequestProperty(
"Accept",
"image/gif,image/jpeg,image/pjpeg,image/pjpeg,"
+"application/x-shockwave-flash,application/xaml+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;icurrentPartSize会是的该线程多写入几个字节,
//但是下一个线程会从文件的指定位置写入,就会覆盖掉之前线程多写的一部分内容
currentPart.close();
inStream.close();
}
catch (Exception e)
{
e.printStackTrace();
}
}
}
}

上面代码的大致思路如下:

DownUtils类—启动、实现多线程下载的类;

DownThread—线程下载类,作为DownUtils类的辅助类,主要实现单个线程的下载逻辑。

程序中DownUtils类中的download()方法负责按如下步骤实现多线程下载

创建URL对象

获取指定URL对象所指向资源的大小(通过getConnectLength()方法获得),此处,用到了URLConnection类,该类代表Java应用程序和URL之间的通信链接。

在本地磁盘上创建一个与网络资源具有同样大小的空文件

计算每个线程应该下载网络资源的哪个部分

依次创建、启动多个线程来下载网络资源的指定部分

创建一个和URL的连接,并发送请求、读取此URL引用的资源需要如下几个步骤:

通过调用URL对象的openConnection()方法来创建URLConnection对象;

设置URLConnection的参数和普通请求属性;

如果只是发送GET方式请求,则使用connect()方法建立和远程资源之间的实际连接即可;如果需要发送POST方式的请求,则需要获取URLConnection实例对应的输出流来发送请求参数

远程资源变为可用,程序可以访问远程资源的头字段或通过输入流读取远程资源的数据

测试函数如下:

public class MultiThreadDown
{
public static void main(String[] args)
throws Exception
{
//初始化DownUtil对象
final DownUtil downUtil=new DownUtil("http://img10.360buyimg.com/n0/jfs"+
"/t18166/359/202833592/24066/9da49/5a628ffeN32d2b7c8.jpg","ios.png",4);
//开始下载
downUtil.download();
new Thread(()->
{
while(downUtil.getCompleteRate()<1)
{
//每隔0.1秒查询一次任务的完成进度
//GUI程序中可根据该进度来绘制进度条
System.out.println("已完成:"+downUtil.getCompleteRate());
try
{
Thread.sleep(100);
}
catch (Exception ex){}
}
}).start();
}
}