Java 多线程断点下载文件_详解
基本原理:利用URLConnection获取要下载文件的长度、头部等相关信息,并设置响应的头部信息。并且通过URLConnection获取输入流,将文件分成指定的块,每一块单独开辟一个线程完成数据的读取、写入。通过输入流读取下载文件的信息,然后将读取的信息用RandomAccessFile随机写入到本地文件中。同时,每个线程写入的数据都文件指针也就是写入数据的长度,需要保存在一个临时文件中。这样当本次下载没有完成的时候,下次下载的时候就从这个文件中读取上一次下载的文件长度,然后继续接着上一次的位置开始下载。并且将本次下载的长度写入到这个文件中。
个人博客:
http://blog.csdn.net/IBM_hoojo
email: hoojo_@126.com
一、下载文件信息类、实体
封装即将下载资源的信息
- package com.hoo.entity;
- /**
- * <b>function:</b> 下载文件信息类
- * @author hoojo
- * @createDate 2011-9-21 下午05:14:58
- * @file DownloadInfo.java
- * @package com.hoo.entity
- * @project MultiThreadDownLoad
- * @blog http://blog.csdn.net/IBM_hoojo
- * @email hoojo_@126.com
- * @version 1.0
- */
- publicclass DownloadInfo {
- //下载文件url
- private String url;
- //下载文件名称
- private String fileName;
- //下载文件路径
- private String filePath;
- //分成多少段下载, 每一段用一个线程完成下载
- privateint splitter;
- //下载文件默认保存路径
- privatefinalstatic String FILE_PATH = "C:/temp";
- //默认分块数、线程数
- privatefinalstaticint SPLITTER_NUM = 5;
- public DownloadInfo() {
- super();
- }
- /**
- * @param url 下载地址
- */
- public DownloadInfo(String url) {
- this(url, null, null, SPLITTER_NUM);
- }
- /**
- * @param url 下载地址url
- * @param splitter 分成多少段或是多少个线程下载
- */
- public DownloadInfo(String url, int splitter) {
- this(url, null, null, splitter);
- }
- /***
- * @param url 下载地址
- * @param fileName 文件名称
- * @param filePath 文件保存路径
- * @param splitter 分成多少段或是多少个线程下载
- */
- public DownloadInfo(String url, String fileName, String filePath, int splitter) {
- super();
- if (url == null || "".equals(url)) {
- throw new RuntimeException("url is not null!");
- }
- this.url = url;
- this.fileName = (fileName == null || "".equals(fileName)) ? getFileName(url) : fileName;
- this.filePath = (filePath == null || "".equals(filePath)) ? FILE_PATH : filePath;
- this.splitter = (splitter < 1) ? SPLITTER_NUM : splitter;
- }
- /**
- * <b>function:</b> 通过url获得文件名称
- * @author hoojo
- * @createDate 2011-9-30 下午05:00:00
- * @param url
- * @return
- */
- private String getFileName(String url) {
- return url.substring(url.lastIndexOf("/") + 1, url.length());
- }
- public String getUrl() {
- return url;
- }
- publicvoid setUrl(String url) {
- if (url == null || "".equals(url)) {
- thrownew RuntimeException("url is not null!");
- }
- this.url = url;
- }
- public String getFileName() {
- return fileName;
- }
- publicvoid setFileName(String fileName) {
- this.fileName = (fileName == null || "".equals(fileName)) ? getFileName(url) : fileName;
- }
- public String getFilePath() {
- return filePath;
- }
- publicvoid setFilePath(String filePath) {
- this.filePath = (filePath == null || "".equals(filePath)) ? FILE_PATH : filePath;
- }
- publicint getSplitter() {
- return splitter;
- }
- publicvoid setSplitter(int splitter) {
- this.splitter = (splitter < 1) ? SPLITTER_NUM : splitter;
- }
- @Override
- public String toString() {
- returnthis.url + "#" + this.fileName + "#" + this.filePath + "#" + this.splitter;
- }
- }
package com.hoo.entity; /** * <b>function:</b> 下载文件信息类 * @author hoojo * @createDate 2011-9-21 下午05:14:58 * @file DownloadInfo.java * @package com.hoo.entity * @project MultiThreadDownLoad * @blog http://blog.csdn.net/IBM_hoojo * @email hoojo_@126.com * @version 1.0 */ public class DownloadInfo { //下载文件url private String url; //下载文件名称 private String fileName; //下载文件路径 private String filePath; //分成多少段下载, 每一段用一个线程完成下载 private int splitter; //下载文件默认保存路径 private final static String FILE_PATH = "C:/temp"; //默认分块数、线程数 private final static int SPLITTER_NUM = 5; public DownloadInfo() { super(); } /** * @param url 下载地址 */ public DownloadInfo(String url) { this(url, null, null, SPLITTER_NUM); } /** * @param url 下载地址url * @param splitter 分成多少段或是多少个线程下载 */ public DownloadInfo(String url, int splitter) { this(url, null, null, splitter); } /*** * @param url 下载地址 * @param fileName 文件名称 * @param filePath 文件保存路径 * @param splitter 分成多少段或是多少个线程下载 */ public DownloadInfo(String url, String fileName, String filePath, int splitter) { super(); if (url == null || "".equals(url)) { throw new RuntimeException("url is not null!"); } this.url = url; this.fileName = (fileName == null || "".equals(fileName)) ? getFileName(url) : fileName; this.filePath = (filePath == null || "".equals(filePath)) ? FILE_PATH : filePath; this.splitter = (splitter < 1) ? SPLITTER_NUM : splitter; } /** * <b>function:</b> 通过url获得文件名称 * @author hoojo * @createDate 2011-9-30 下午05:00:00 * @param url * @return */ private String getFileName(String url) { return url.substring(url.lastIndexOf("/") + 1, url.length()); } public String getUrl() { return url; } public void setUrl(String url) { if (url == null || "".equals(url)) { throw new RuntimeException("url is not null!"); } this.url = url; } public String getFileName() { return fileName; } public void setFileName(String fileName) { this.fileName = (fileName == null || "".equals(fileName)) ? getFileName(url) : fileName; } public String getFilePath() { return filePath; } public void setFilePath(String filePath) { this.filePath = (filePath == null || "".equals(filePath)) ? FILE_PATH : filePath; } public int getSplitter() { return splitter; } public void setSplitter(int splitter) { this.splitter = (splitter < 1) ? SPLITTER_NUM : splitter; } @Override public String toString() { return this.url + "#" + this.fileName + "#" + this.filePath + "#" + this.splitter; } }
二、随机写入一段文件
- package com.hoo.download;
- import java.io.IOException;
- import java.io.RandomAccessFile;
- /**
- * <b>function:</b> 写入文件、保存文件
- * @author hoojo
- * @createDate 2011-9-21 下午05:44:02
- * @file SaveItemFile.java
- * @package com.hoo.download
- * @project MultiThreadDownLoad
- * @blog http://blog.csdn.net/IBM_hoojo
- * @email hoojo_@126.com
- * @version 1.0
- */
- publicclass SaveItemFile {
- //存储文件
- private RandomAccessFile itemFile;
- public SaveItemFile() throws IOException {
- this("", 0);
- }
- /**
- * @param name 文件路径、名称
- * @param pos 写入点位置 position
- * @throws IOException
- */
- public SaveItemFile(String name, long pos) throws IOException {
- itemFile = new RandomAccessFile(name, "rw");
- //在指定的pos位置开始写入数据
- itemFile.seek(pos);
- }
- /**
- * <b>function:</b> 同步方法写入文件
- * @author hoojo
- * @createDate 2011-9-26 下午12:21:22
- * @param buff 缓冲数组
- * @param start 起始位置
- * @param length 长度
- * @return
- */
- publicsynchronizedint write(byte[] buff, int start, int length) {
- int i = -1;
- try {
- itemFile.write(buff, start, length);
- i = length;
- } catch (IOException e) {
- e.printStackTrace();
- }
- return i;
- }
- publicvoid close() throws IOException {
- if (itemFile != null) {
- itemFile.close();
- }
- }
- }
package com.hoo.download; import java.io.IOException; import java.io.RandomAccessFile; /** * <b>function:</b> 写入文件、保存文件 * @author hoojo * @createDate 2011-9-21 下午05:44:02 * @file SaveItemFile.java * @package com.hoo.download * @project MultiThreadDownLoad * @blog http://blog.csdn.net/IBM_hoojo * @email hoojo_@126.com * @version 1.0 */ public class SaveItemFile { //存储文件 private RandomAccessFile itemFile; public SaveItemFile() throws IOException { this("", 0); } /** * @param name 文件路径、名称 * @param pos 写入点位置 position * @throws IOException */ public SaveItemFile(String name, long pos) throws IOException { itemFile = new RandomAccessFile(name, "rw"); //在指定的pos位置开始写入数据 itemFile.seek(pos); } /** * <b>function:</b> 同步方法写入文件 * @author hoojo * @createDate 2011-9-26 下午12:21:22 * @param buff 缓冲数组 * @param start 起始位置 * @param length 长度 * @return */ public synchronized int write(byte[] buff, int start, int length) { int i = -1; try { itemFile.write(buff, start, length); i = length; } catch (IOException e) { e.printStackTrace(); } return i; } public void close() throws IOException { if (itemFile != null) { itemFile.close(); } } }
这个类主要是完成向本地的指定文件指针出开始写入文件,并返回当前写入文件的长度(文件指针)。这个类将被线程调用,文件被分成对应的块后,将被线程调用。每个线程都将会调用这个类完成文件的随机写入。
三、单个线程下载文件
- package com.hoo.download;
- import java.io.IOException;
- import java.io.InputStream;
- import java.net.HttpURLConnection;
- import java.net.MalformedURLException;
- import java.net.URL;
- import java.net.URLConnection;
- import com.hoo.util.LogUtils;
- /**
- * <b>function:</b> 单线程下载文件
- * @author hoojo
- * @createDate 2011-9-22 下午02:55:10
- * @file DownloadFile.java
- * @package com.hoo.download
- * @project MultiThreadDownLoad
- * @blog http://blog.csdn.net/IBM_hoojo
- * @email hoojo_@126.com
- * @version 1.0
- */
- publicclass DownloadFile extends Thread {
- //下载文件url
- private String url;
- //下载文件起始位置
- privatelong startPos;
- //下载文件结束位置
- privatelong endPos;
- //线程id
- privateint threadId;
- //下载是否完成
- privateboolean isDownloadOver = false;
- private SaveItemFile itemFile;
- privatestaticfinalint BUFF_LENGTH = 1024 * 8;
- /**
- * @param url 下载文件url
- * @param name 文件名称
- * @param startPos 下载文件起点
- * @param endPos 下载文件结束点
- * @param threadId 线程id
- * @throws IOException
- */
- public DownloadFile(String url, String name, long startPos, long endPos, int threadId) throws IOException {
- super();
- this.url = url;
- this.startPos = startPos;
- this.endPos = endPos;
- this.threadId = threadId;
- //分块下载写入文件内容
- this.itemFile = new SaveItemFile(name, startPos);
- }
- @Override
- publicvoid run() {
- while (endPos > startPos && !isDownloadOver) {
- try {
- URL url = new URL(this.url);
- HttpURLConnection conn = (HttpURLConnection) url.openConnection();
- // 设置连接超时时间为10000ms
- conn.setConnectTimeout(10000);
- // 设置读取数据超时时间为10000ms
- conn.setReadTimeout(10000);
- setHeader(conn);
- String property = "bytes=" + startPos + "-";
- conn.setRequestProperty("RANGE", property);
- //输出log信息
- LogUtils.log("开始 " + threadId + ":" + property + endPos);
- //printHeader(conn);
- //获取文件输入流,读取文件内容
- InputStream is = conn.getInputStream();
- byte[] buff = newbyte[BUFF_LENGTH];
- int length = -1;
- LogUtils.log("#start#Thread: " + threadId + ", startPos: " + startPos + ", endPos: " + endPos);
- while ((length = is.read(buff)) > 0 && startPos < endPos && !isDownloadOver) {
- //写入文件内容,返回最后写入的长度
- startPos += itemFile.write(buff, 0, length);
- }
- LogUtils.log("#over#Thread: " + threadId + ", startPos: " + startPos + ", endPos: " + endPos);
- LogUtils.log("Thread " + threadId + " is execute over!");
- this.isDownloadOver = true;
- } catch (MalformedURLException e) {
- e.printStackTrace();
- } catch (IOException e) {
- e.printStackTrace();
- } finally {
- try {
- if (itemFile != null) {
- itemFile.close();
- }
- } catch (IOException e) {
- e.printStackTrace();
- }
- }
- }
- if (endPos < startPos && !isDownloadOver) {
- LogUtils.log("Thread " + threadId + " startPos > endPos, not need download file !");
- this.isDownloadOver = true;
- }
- if (endPos == startPos && !isDownloadOver) {
- LogUtils.log("Thread " + threadId + " startPos = endPos, not need download file !");
- this.isDownloadOver = true;
- }
- }
- /**
- * <b>function:</b> 打印下载文件头部信息
- * @author hoojo
- * @createDate 2011-9-22 下午05:44:35
- * @param conn HttpURLConnection
- */
- publicstaticvoid printHeader(URLConnection conn) {
- int i = 1;
- while (true) {
- String header = conn.getHeaderFieldKey(i);
- i++;
- if (header != null) {
- LogUtils.info(header + ":" + conn.getHeaderField(i));
- } else {
- break;
- }
- }
- }
- /**
- * <b>function:</b> 设置URLConnection的头部信息,伪装请求信息
- * @author hoojo
- * @createDate 2011-9-28 下午05:29:43
- * @param con
- */
- publicstaticvoid setHeader(URLConnection conn) {
- conn.setRequestProperty("User-Agent", "Mozilla/5.0 (X11; U; Linux i686; en-US; rv:1.9.0.3) Gecko/2008092510 Ubuntu/8.04 (hardy) Firefox/3.0.3");
- conn.setRequestProperty("Accept-Language", "en-us,en;q=0.7,zh-cn;q=0.3");
- conn.setRequestProperty("Accept-Encoding", "utf-8");
- conn.setRequestProperty("Accept-Charset", "ISO-8859-1,utf-8;q=0.7,*;q=0.7");
- conn.setRequestProperty("Keep-Alive", "300");
- conn.setRequestProperty("connnection", "keep-alive");
- conn.setRequestProperty("If-Modified-Since", "Fri, 02 Jan 2009 17:00:05 GMT");
- conn.setRequestProperty("If-None-Match", "\"1261d8-4290-df64d224\"");
- conn.setRequestProperty("Cache-conntrol", "max-age=0");
- conn.setRequestProperty("Referer", "http://www.baidu.com");
- }
- publicboolean isDownloadOver() {
- return isDownloadOver;
- }
- publiclong getStartPos() {
- return startPos;
- }
- publiclong getEndPos() {
- return endPos;
- }
- }
package com.hoo.download; import java.io.IOException; import java.io.InputStream; import java.net.HttpURLConnection; import java.net.MalformedURLException; import java.net.URL; import java.net.URLConnection; import com.hoo.util.LogUtils; /** * <b>function:</b> 单线程下载文件 * @author hoojo * @createDate 2011-9-22 下午02:55:10 * @file DownloadFile.java * @package com.hoo.download * @project MultiThreadDownLoad * @blog http://blog.csdn.net/IBM_hoojo * @email hoojo_@126.com * @version 1.0 */ public class DownloadFile extends Thread { //下载文件url private String url; //下载文件起始位置 private long startPos; //下载文件结束位置 private long endPos; //线程id private int threadId; //下载是否完成 private boolean isDownloadOver = false; private SaveItemFile itemFile; private static final int BUFF_LENGTH = 1024 * 8; /** * @param url 下载文件url * @param name 文件名称 * @param startPos 下载文件起点 * @param endPos 下载文件结束点 * @param threadId 线程id * @throws IOException */ public DownloadFile(String url, String name, long startPos, long endPos, int threadId) throws IOException { super(); this.url = url; this.startPos = startPos; this.endPos = endPos; this.threadId = threadId; //分块下载写入文件内容 this.itemFile = new SaveItemFile(name, startPos); } @Override public void run() { while (endPos > startPos && !isDownloadOver) { try { URL url = new URL(this.url); HttpURLConnection conn = (HttpURLConnection) url.openConnection(); // 设置连接超时时间为10000ms conn.setConnectTimeout(10000); // 设置读取数据超时时间为10000ms conn.setReadTimeout(10000); setHeader(conn); String property = "bytes=" + startPos + "-"; conn.setRequestProperty("RANGE", property); //输出log信息 LogUtils.log("开始 " + threadId + ":" + property + endPos); //printHeader(conn); //获取文件输入流,读取文件内容 InputStream is = conn.getInputStream(); byte[] buff = new byte[BUFF_LENGTH]; int length = -1; LogUtils.log("#start#Thread: " + threadId + ", startPos: " + startPos + ", endPos: " + endPos); while ((length = is.read(buff)) > 0 && startPos < endPos && !isDownloadOver) { //写入文件内容,返回最后写入的长度 startPos += itemFile.write(buff, 0, length); } LogUtils.log("#over#Thread: " + threadId + ", startPos: " + startPos + ", endPos: " + endPos); LogUtils.log("Thread " + threadId + " is execute over!"); this.isDownloadOver = true; } catch (MalformedURLException e) { e.printStackTrace(); } catch (IOException e) { e.printStackTrace(); } finally { try { if (itemFile != null) { itemFile.close(); } } catch (IOException e) { e.printStackTrace(); } } } if (endPos < startPos && !isDownloadOver) { LogUtils.log("Thread " + threadId + " startPos > endPos, not need download file !"); this.isDownloadOver = true; } if (endPos == startPos && !isDownloadOver) { LogUtils.log("Thread " + threadId + " startPos = endPos, not need download file !"); this.isDownloadOver = true; } } /** * <b>function:</b> 打印下载文件头部信息 * @author hoojo * @createDate 2011-9-22 下午05:44:35 * @param conn HttpURLConnection */ public static void printHeader(URLConnection conn) { int i = 1; while (true) { String header = conn.getHeaderFieldKey(i); i++; if (header != null) { LogUtils.info(header + ":" + conn.getHeaderField(i)); } else { break; } } } /** * <b>function:</b> 设置URLConnection的头部信息,伪装请求信息 * @author hoojo * @createDate 2011-9-28 下午05:29:43 * @param con */ public static void setHeader(URLConnection conn) { conn.setRequestProperty("User-Agent", "Mozilla/5.0 (X11; U; Linux i686; en-US; rv:1.9.0.3) Gecko/2008092510 Ubuntu/8.04 (hardy) Firefox/3.0.3"); conn.setRequestProperty("Accept-Language", "en-us,en;q=0.7,zh-cn;q=0.3"); conn.setRequestProperty("Accept-Encoding", "utf-8"); conn.setRequestProperty("Accept-Charset", "ISO-8859-1,utf-8;q=0.7,*;q=0.7"); conn.setRequestProperty("Keep-Alive", "300"); conn.setRequestProperty("connnection", "keep-alive"); conn.setRequestProperty("If-Modified-Since", "Fri, 02 Jan 2009 17:00:05 GMT"); conn.setRequestProperty("If-None-Match", "\"1261d8-4290-df64d224\""); conn.setRequestProperty("Cache-conntrol", "max-age=0"); conn.setRequestProperty("Referer", "http://www.baidu.com"); } public boolean isDownloadOver() { return isDownloadOver; } public long getStartPos() { return startPos; } public long getEndPos() { return endPos; } }
这个类主要是完成单个线程的文件下载,将通过URLConnection读取指定url的资源信息。然后用InputStream读取文件内容,然后调用调用SaveItemFile类,向本地写入当前要读取的块的内容。
四、分段多线程写入文件内容
- package com.hoo.download;
- import java.io.DataInputStream;
- import java.io.DataOutputStream;
- import java.io.File;
- import java.io.FileInputStream;
- import java.io.FileOutputStream;
- import java.io.IOException;
- import java.net.HttpURLConnection;
- import java.net.MalformedURLException;
- import java.net.URL;
- import com.hoo.entity.DownloadInfo;
- import com.hoo.util.LogUtils;
- /**
- * <b>function:</b> 分批量下载文件
- * @author hoojo
- * @createDate 2011-9-22 下午05:51:54
- * @file BatchDownloadFile.java
- * @package com.hoo.download
- * @project MultiThreadDownLoad
- * @blog http://blog.csdn.net/IBM_hoojo
- * @email hoojo_@126.com
- * @version 1.0
- */
- publicclass BatchDownloadFile implements Runnable {
- //下载文件信息
- private DownloadInfo downloadInfo;
- //一组开始下载位置
- privatelong[] startPos;
- //一组结束下载位置
- privatelong[] endPos;
- //休眠时间
- privatestaticfinalint SLEEP_SECONDS = 500;
- //子线程下载
- private DownloadFile[] fileItem;
- //文件长度
- privateint length;
- //是否第一个文件
- privateboolean first = true;
- //是否停止下载
- privateboolean stop = false;
- //临时文件信息
- private File tempFile;
- public BatchDownloadFile(DownloadInfo downloadInfo) {
- this.downloadInfo = downloadInfo;
- String tempPath = this.downloadInfo.getFilePath() + File.separator + downloadInfo.getFileName() + ".position";
- tempFile = new File(tempPath);
- //如果存在读入点位置的文件
- if (tempFile.exists()) {
- first = false;
- //就直接读取内容
- try {
- readPosInfo();
- } catch (IOException e) {
- e.printStackTrace();
- }
- } else {
- //数组的长度就要分成多少段的数量
- startPos = newlong[downloadInfo.getSplitter()];
- endPos = newlong[downloadInfo.getSplitter()];
- }
- }
- @Override
- publicvoid run() {
- //首次下载,获取下载文件长度
- if (first) {
- length = this.getFileSize();//获取文件长度
- if (length == -1) {
- LogUtils.log("file length is know!");
- stop = true;
- } elseif (length == -2) {
- LogUtils.log("read file length is error!");
- stop = true;
- } else if (length > 0) {
- /**
- * eg
- * start: 1, 3, 5, 7, 9
- * end: 3, 5, 7, 9, length
- */
- for (int i = 0, len = startPos.length; i < len; i++) {
- int size = i * (length / len);
- startPos[i] = size;
- //设置最后一个结束点的位置
- if (i == len - 1) {
- endPos[i] = length;
- } else {
- size = (i + 1) * (length / len);
- endPos[i] = size;
- }
- LogUtils.log("start-end Position[" + i + "]: " + startPos[i] + "-" + endPos[i]);
- }
- } else {
- LogUtils.log("get file length is error, download is stop!");
- stop = true;
- }
- }
- //子线程开始下载
- if (!stop) {
- //创建单线程下载对象数组
- fileItem = new DownloadFile[startPos.length];//startPos.length = downloadInfo.getSplitter()
- for (int i = 0; i < startPos.length; i++) {
- try {
- //创建指定个数单线程下载对象,每个线程独立完成指定块内容的下载
- fileItem[i] = new DownloadFile(
- downloadInfo.getUrl(),
- this.downloadInfo.getFilePath() + File.separator + downloadInfo.getFileName(),
- startPos[i], endPos[i], i
- );
- fileItem[i].start();//启动线程,开始下载
- LogUtils.log("Thread: " + i + ", startPos: " + startPos[i] + ", endPos: " + endPos[i]);
- } catch (IOException e) {
- e.printStackTrace();
- }
- }
- //循环写入下载文件长度信息
- while (!stop) {
- try {
- writePosInfo();
- LogUtils.log("downloading……");
- Thread.sleep(SLEEP_SECONDS);
- stop = true;
- } catch (IOException e) {
- e.printStackTrace();
- } catch (InterruptedException e) {
- e.printStackTrace();
- }
- for (int i = 0; i < startPos.length; i++) {
- if (!fileItem[i].isDownloadOver()) {
- stop = false;
- break;
- }
- }
- }
- LogUtils.info("Download task is finished!");
- }
- }
- /**
- * 将写入点数据保存在临时文件中
- * @author hoojo
- * @createDate 2011-9-23 下午05:25:37
- * @throws IOException
- */
- privatevoid writePosInfo() throws IOException {
- DataOutputStream dos = new DataOutputStream(new FileOutputStream(tempFile));
- dos.writeInt(startPos.length);
- for (int i = 0; i < startPos.length; i++) {
- dos.writeLong(fileItem[i].getStartPos());
- dos.writeLong(fileItem[i].getEndPos());
- //LogUtils.info("[" + fileItem[i].getStartPos() + "#" + fileItem[i].getEndPos() + "]");
- }
- dos.close();
- }
- /**
- * <b>function:</b>读取写入点的位置信息
- * @author hoojo
- * @createDate 2011-9-23 下午05:30:29
- * @throws IOException
- */
- privatevoid readPosInfo() throws IOException {
- DataInputStream dis = new DataInputStream(new FileInputStream(tempFile));
- int startPosLength = dis.readInt();
- startPos = newlong[startPosLength];
- endPos = new long[startPosLength];
- for (int i = 0; i < startPosLength; i++) {
- startPos[i] = dis.readLong();
- endPos[i] = dis.readLong();
- }
- dis.close();
- }
- /**
- * <b>function:</b> 获取下载文件的长度
- * @author hoojo
- * @createDate 2011-9-26 下午12:15:08
- * @return
- */
- privateint getFileSize() {
- int fileLength = -1;
- try {
- URL url = new URL(this.downloadInfo.getUrl());
- HttpURLConnection conn = (HttpURLConnection) url.openConnection();
- DownloadFile.setHeader(conn);
- int stateCode = conn.getResponseCode();
- //判断http status是否为HTTP/1.1 206 Partial Content或者200 OK
- if (stateCode != HttpURLConnection.HTTP_OK && stateCode != HttpURLConnection.HTTP_PARTIAL) {
- LogUtils.log("Error Code: " + stateCode);
- return -2;
- } elseif (stateCode >= 400) {
- LogUtils.log("Error Code: " + stateCode);
- return -2;
- } else {
- //获取长度
- fileLength = conn.getContentLength();
- LogUtils.log("FileLength: " + fileLength);
- }
- //读取文件长度
- /*for (int i = 1; ; i++) {
- String header = conn.getHeaderFieldKey(i);
- if (header != null) {
- if ("Content-Length".equals(header)) {
- fileLength = Integer.parseInt(conn.getHeaderField(i));
- break;
- }
- } else {
- break;
- }
- }
- */
- DownloadFile.printHeader(conn);
- } catch (MalformedURLException e) {
- e.printStackTrace();
- } catch (IOException e) {
- e.printStackTrace();
- }
- return fileLength;
- }
- }
package com.hoo.download; import java.io.DataInputStream; import java.io.DataOutputStream; import java.io.File; import java.io.FileInputStream; import java.io.FileOutputStream; import java.io.IOException; import java.net.HttpURLConnection; import java.net.MalformedURLException; import java.net.URL; import com.hoo.entity.DownloadInfo; import com.hoo.util.LogUtils; /** * <b>function:</b> 分批量下载文件 * @author hoojo * @createDate 2011-9-22 下午05:51:54 * @file BatchDownloadFile.java * @package com.hoo.download * @project MultiThreadDownLoad * @blog http://blog.csdn.net/IBM_hoojo * @email hoojo_@126.com * @version 1.0 */ public class BatchDownloadFile implements Runnable { //下载文件信息 private DownloadInfo downloadInfo; //一组开始下载位置 private long[] startPos; //一组结束下载位置 private long[] endPos; //休眠时间 private static final int SLEEP_SECONDS = 500; //子线程下载 private DownloadFile[] fileItem; //文件长度 private int length; //是否第一个文件 private boolean first = true; //是否停止下载 private boolean stop = false; //临时文件信息 private File tempFile; public BatchDownloadFile(DownloadInfo downloadInfo) { this.downloadInfo = downloadInfo; String tempPath = this.downloadInfo.getFilePath() + File.separator + downloadInfo.getFileName() + ".position"; tempFile = new File(tempPath); //如果存在读入点位置的文件 if (tempFile.exists()) { first = false; //就直接读取内容 try { readPosInfo(); } catch (IOException e) { e.printStackTrace(); } } else { //数组的长度就要分成多少段的数量 startPos = new long[downloadInfo.getSplitter()]; endPos = new long[downloadInfo.getSplitter()]; } } @Override public void run() { //首次下载,获取下载文件长度 if (first) { length = this.getFileSize();//获取文件长度 if (length == -1) { LogUtils.log("file length is know!"); stop = true; } else if (length == -2) { LogUtils.log("read file length is error!"); stop = true; } else if (length > 0) { /** * eg * start: 1, 3, 5, 7, 9 * end: 3, 5, 7, 9, length */ for (int i = 0, len = startPos.length; i < len; i++) { int size = i * (length / len); startPos[i] = size; //设置最后一个结束点的位置 if (i == len - 1) { endPos[i] = length; } else { size = (i + 1) * (length / len); endPos[i] = size; } LogUtils.log("start-end Position[" + i + "]: " + startPos[i] + "-" + endPos[i]); } } else { LogUtils.log("get file length is error, download is stop!"); stop = true; } } //子线程开始下载 if (!stop) { //创建单线程下载对象数组 fileItem = new DownloadFile[startPos.length];//startPos.length = downloadInfo.getSplitter() for (int i = 0; i < startPos.length; i++) { try { //创建指定个数单线程下载对象,每个线程独立完成指定块内容的下载 fileItem[i] = new DownloadFile( downloadInfo.getUrl(), this.downloadInfo.getFilePath() + File.separator + downloadInfo.getFileName(), startPos[i], endPos[i], i ); fileItem[i].start();//启动线程,开始下载 LogUtils.log("Thread: " + i + ", startPos: " + startPos[i] + ", endPos: " + endPos[i]); } catch (IOException e) { e.printStackTrace(); } } //循环写入下载文件长度信息 while (!stop) { try { writePosInfo(); LogUtils.log("downloading……"); Thread.sleep(SLEEP_SECONDS); stop = true; } catch (IOException e) { e.printStackTrace(); } catch (InterruptedException e) { e.printStackTrace(); } for (int i = 0; i < startPos.length; i++) { if (!fileItem[i].isDownloadOver()) { stop = false; break; } } } LogUtils.info("Download task is finished!"); } } /** * 将写入点数据保存在临时文件中 * @author hoojo * @createDate 2011-9-23 下午05:25:37 * @throws IOException */ private void writePosInfo() throws IOException { DataOutputStream dos = new DataOutputStream(new FileOutputStream(tempFile)); dos.writeInt(startPos.length); for (int i = 0; i < startPos.length; i++) { dos.writeLong(fileItem[i].getStartPos()); dos.writeLong(fileItem[i].getEndPos()); //LogUtils.info("[" + fileItem[i].getStartPos() + "#" + fileItem[i].getEndPos() + "]"); } dos.close(); } /** * <b>function:</b>读取写入点的位置信息 * @author hoojo * @createDate 2011-9-23 下午05:30:29 * @throws IOException */ private void readPosInfo() throws IOException { DataInputStream dis = new DataInputStream(new FileInputStream(tempFile)); int startPosLength = dis.readInt(); startPos = new long[startPosLength]; endPos = new long[startPosLength]; for (int i = 0; i < startPosLength; i++) { startPos[i] = dis.readLong(); endPos[i] = dis.readLong(); } dis.close(); } /** * <b>function:</b> 获取下载文件的长度 * @author hoojo * @createDate 2011-9-26 下午12:15:08 * @return */ private int getFileSize() { int fileLength = -1; try { URL url = new URL(this.downloadInfo.getUrl()); HttpURLConnection conn = (HttpURLConnection) url.openConnection(); DownloadFile.setHeader(conn); int stateCode = conn.getResponseCode(); //判断http status是否为HTTP/1.1 206 Partial Content或者200 OK if (stateCode != HttpURLConnection.HTTP_OK && stateCode != HttpURLConnection.HTTP_PARTIAL) { LogUtils.log("Error Code: " + stateCode); return -2; } else if (stateCode >= 400) { LogUtils.log("Error Code: " + stateCode); return -2; } else { //获取长度 fileLength = conn.getContentLength(); LogUtils.log("FileLength: " + fileLength); } //读取文件长度 /*for (int i = 1; ; i++) { String header = conn.getHeaderFieldKey(i); if (header != null) { if ("Content-Length".equals(header)) { fileLength = Integer.parseInt(conn.getHeaderField(i)); break; } } else { break; } } */ DownloadFile.printHeader(conn); } catch (MalformedURLException e) { e.printStackTrace(); } catch (IOException e) { e.printStackTrace(); } return fileLength; } }
这个类主要是完成读取指定url资源的内容,获取该资源的长度。然后将该资源分成指定的块数,将每块的起始下载位置、结束下载位置,分别保存在一个数组中。每块都单独开辟一个独立线程开始下载。在开始下载之前,需要创建一个临时文件,写入当前下载线程的开始下载指针位置和结束下载指针位置。
五、工具类、测试类
日志工具类
- package com.hoo.util;
- /**
- * <b>function:</b> 日志工具类
- * @author hoojo
- * @createDate 2011-9-21 下午05:21:27
- * @file LogUtils.java
- * @package com.hoo.util
- * @project MultiThreadDownLoad
- * @blog http://blog.csdn.net/IBM_hoojo
- * @email hoojo_@126.com
- * @version 1.0
- */
- publicabstractclass LogUtils {
- publicstaticvoid log(Object message) {
- System.err.println(message);
- }
- publicstaticvoid log(String message) {
- System.err.println(message);
- }
- publicstaticvoid log(int message) {
- System.err.println(message);
- }
- publicstaticvoid info(Object message) {
- System.out.println(message);
- }
- publicstaticvoid info(String message) {
- System.out.println(message);
- }
- publicstaticvoid info(int message) {
- System.out.println(message);
- }
- }
package com.hoo.util; /** * <b>function:</b> 日志工具类 * @author hoojo * @createDate 2011-9-21 下午05:21:27 * @file LogUtils.java * @package com.hoo.util * @project MultiThreadDownLoad * @blog http://blog.csdn.net/IBM_hoojo * @email hoojo_@126.com * @version 1.0 */ public abstract class LogUtils { public static void log(Object message) { System.err.println(message); } public static void log(String message) { System.err.println(message); } public static void log(int message) { System.err.println(message); } public static void info(Object message) { System.out.println(message); } public static void info(String message) { System.out.println(message); } public static void info(int message) { System.out.println(message); } }
下载工具类
- package com.hoo.util;
- import com.hoo.download.BatchDownloadFile;
- import com.hoo.entity.DownloadInfo;
- /**
- * <b>function:</b> 分块多线程下载工具类
- * @author hoojo
- * @createDate 2011-9-28 下午05:22:18
- * @file DownloadUtils.java
- * @package com.hoo.util
- * @project MultiThreadDownLoad
- * @blog http://blog.csdn.net/IBM_hoojo
- * @email hoojo_@126.com
- * @version 1.0
- */
- publicabstractclass DownloadUtils {
- publicstaticvoid download(String url) {
- DownloadInfo bean = new DownloadInfo(url);
- LogUtils.info(bean);
- BatchDownloadFile down = new BatchDownloadFile(bean);
- new Thread(down).start();
- }
- publicstaticvoid download(String url, int threadNum) {
- DownloadInfo bean = new DownloadInfo(url, threadNum);
- LogUtils.info(bean);
- BatchDownloadFile down = new BatchDownloadFile(bean);
- new Thread(down).start();
- }
- publicstaticvoid download(String url, String fileName, String filePath, int threadNum) {
- DownloadInfo bean = new DownloadInfo(url, fileName, filePath, threadNum);
- LogUtils.info(bean);
- BatchDownloadFile down = new BatchDownloadFile(bean);
- new Thread(down).start();
- }
- }
package com.hoo.util; import com.hoo.download.BatchDownloadFile; import com.hoo.entity.DownloadInfo; /** * <b>function:</b> 分块多线程下载工具类 * @author hoojo * @createDate 2011-9-28 下午05:22:18 * @file DownloadUtils.java * @package com.hoo.util * @project MultiThreadDownLoad * @blog http://blog.csdn.net/IBM_hoojo * @email hoojo_@126.com * @version 1.0 */ public abstract class DownloadUtils { public static void download(String url) { DownloadInfo bean = new DownloadInfo(url); LogUtils.info(bean); BatchDownloadFile down = new BatchDownloadFile(bean); new Thread(down).start(); } public static void download(String url, int threadNum) { DownloadInfo bean = new DownloadInfo(url, threadNum); LogUtils.info(bean); BatchDownloadFile down = new BatchDownloadFile(bean); new Thread(down).start(); } public static void download(String url, String fileName, String filePath, int threadNum) { DownloadInfo bean = new DownloadInfo(url, fileName, filePath, threadNum); LogUtils.info(bean); BatchDownloadFile down = new BatchDownloadFile(bean); new Thread(down).start(); } }
下载测试类
- package com.hoo.test;
- import com.hoo.util.DownloadUtils;
- /**
- * <b>function:</b> 下载测试
- * @author hoojo
- * @createDate 2011-9-23 下午05:49:46
- * @file TestDownloadMain.java
- * @package com.hoo.download
- * @project MultiThreadDownLoad
- * @blog http://blog.csdn.net/IBM_hoojo
- * @email hoojo_@126.com
- * @version 1.0
- */
- publicclass TestDownloadMain {
- publicstaticvoid main(String[] args) {
- /*DownloadInfo bean = new DownloadInfo("http://i7.meishichina.com/Health/UploadFiles/201109/2011092116224363.jpg");
- System.out.println(bean);
- BatchDownloadFile down = new BatchDownloadFile(bean);
- new Thread(down).start();*/
- //DownloadUtils.download("http://i7.meishichina.com/Health/UploadFiles/201109/2011092116224363.jpg");
- DownloadUtils.download("http://mp3.baidu.com/j?j=2&url=http%3A%2F%2Fzhangmenshiting2.baidu.com%2Fdata%2Fmusic%2F1669425%2F%25E9%2599%25B7%25E5%2585%25A5%25E7%2588%25B1%25E9%2587%258C%25E9%259D%25A2.mp3%3Fxcode%3D2ff36fb70737c816553396c56deab3f1", "aa.mp3", "c:/temp", 5);
- }
- }
package com.hoo.test; import com.hoo.util.DownloadUtils; /** * <b>function:</b> 下载测试 * @author hoojo * @createDate 2011-9-23 下午05:49:46 * @file TestDownloadMain.java * @package com.hoo.download * @project MultiThreadDownLoad * @blog http://blog.csdn.net/IBM_hoojo * @email hoojo_@126.com * @version 1.0 */ public class TestDownloadMain { public static void main(String[] args) { /*DownloadInfo bean = new DownloadInfo("http://i7.meishichina.com/Health/UploadFiles/201109/2011092116224363.jpg"); System.out.println(bean); BatchDownloadFile down = new BatchDownloadFile(bean); new Thread(down).start();*/ //DownloadUtils.download("http://i7.meishichina.com/Health/UploadFiles/201109/2011092116224363.jpg"); DownloadUtils.download("http://mp3.baidu.com/j?j=2&url=http%3A%2F%2Fzhangmenshiting2.baidu.com%2Fdata%2Fmusic%2F1669425%2F%25E9%2599%25B7%25E5%2585%25A5%25E7%2588%25B1%25E9%2587%258C%25E9%259D%25A2.mp3%3Fxcode%3D2ff36fb70737c816553396c56deab3f1", "aa.mp3", "c:/temp", 5); } }
多线程下载主要在第三部和第四部,其他的地方还是很好理解。源码中提供相应的注释了,便于理解。
本文转载于:http://blog.csdn.net/ibm_hoojo/article/details/6838222