在Java中使用多线程结合断点续传实现一个简单的文件下载器
2016-04-23 13:25
816 查看
这篇博客介绍在android中使用多线程和断点续传实现一个简单的文件下载器
第一步:启动Tomcat服务器,将需要下载的文件部署到Tomcat服务器上
第二步:使用eclipse创建一个Java工程,并且在工程中添加下面的代码
package com.fyt.multidownload;
import java.io.BufferedReader;
import java.io.File;
import java.io.FileInputStream;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.io.RandomAccessFile;
import java.net.HttpURLConnection;
import java.net.URL;
public class MultiDownload {
//下载文件时开启的线程的个数
static int ThreadCount = 3;
//结束的线程的个数
static int finishedThread = 0;
//设置文件的下载地址
static String path = "http://192.168.0.101:8080/app/genymotion-2.2.0-vbox.exe";
public static void main(String[] args) {
try {
//将下载地址封装成Url对象
URL url = new URL(path);
//创建连接对象,此时未建立连接
HttpURLConnection conn = (HttpURLConnection) url.openConnection();
//设置请求方式为get请求
conn.setRequestMethod("GET");
//设置连接超时
conn.setConnectTimeout(5000);
//设置读取超时
conn.setReadTimeout(5000);
//如果请求发送成功
if(conn.getResponseCode() == 200) {
//获得需要下载的文件的长度
int length = conn.getContentLength();
//创建File对象
File file = new File("genymotion-2.2.0-vbox.exe");
//生成临时文件(临时文件的大小为下载的文件的大小)
RandomAccessFile raf = new RandomAccessFile(file, "rwd");
//设置临时文件的大小
raf.setLength(length);
//关闭临时文件
raf.close();
//计算出每个线程应该下载多少字节
int size = length / ThreadCount;
//遍历创建好的线程
for (int i = 0; i < ThreadCount; i++) {
//计算线程下载的开始位置
int startIndex = i * size;
//计算线程下载的结束位置
int endIndex = (i + 1) * size - 1;
//如果是最后一个线程,那么结束位置写死
if(i == ThreadCount - 1)
{
endIndex = length - 1;
}
//System.out.println("线程" + i + "的下载区间是:" + startIndex + "---" + endIndex);
//创建下载线程
//第一个参数:下载开始的位置
//第二个参数:下载结束的位置
//第三个参数:线程的Id
DownLoadThread thread = new DownLoadThread(startIndex, endIndex, i);
//启动下载线程
thread.start();
}
}
} catch (Exception e) {
e.printStackTrace();
}
}
}
//创建一个继承自线程类的下载线程类
class DownLoadThread extends Thread {
//线程下载的开始位置
int startIndex;
//线程下载的结束位置
int endIndex;
//线程的Id
int threadId;
//下载线程的构造方法
public DownLoadThread(int startIndex, int endIndex, int threadId) {
super();
//设置线程下载的开始位置
this.startIndex = startIndex;
//设置线程下载的结束位置
this.endIndex = endIndex;
//设置线程的Id
this.threadId = threadId;
}
//执行线程
@Override
public void run() {
try {
//创建保存下载进度的临时文件
File progressFile = new File(threadId + ".txt");
//如果进度临时文件存在
if(progressFile.exists()) {
//创建文件输入流
FileInputStream fis = new FileInputStream(progressFile);
//InputStreamReader:创建文件的输入流缓冲区对象
//BufferedReader:创建文件的读取缓冲区对象
BufferedReader br = new BufferedReader(new InputStreamReader(fis));
//从进度临时文件中读取出上一次下载的总进度
//然后与原本的开始位置相加,得到新的开始位置
startIndex += Integer.parseInt(br.readLine());
//关闭文件输入流
fis.close();
}
//System.out.println("线程" + threadId + "的下载区间是:" + startIndex + "---" + endIndex);
//使用URL对象封装下载文件的地址
URL url = new URL(MultiDownload.path);
//创建连接对象,此时未建立连接
HttpURLConnection conn = (HttpURLConnection) url.openConnection();
//设置请求方式为get请求
conn.setRequestMethod("GET");
//设置连接超时
conn.setConnectTimeout(5000);
//设置读取超时
conn.setReadTimeout(5000);
//设置本次http请求所请求的数据的区间
conn.setRequestProperty("Range", "bytes=" + startIndex + "-" + endIndex);
//请求部分数据
//请求部分数据请求成功的响应码是206
if(conn.getResponseCode() == 206) {
//流里此时只有1/3原文件的数据
//从响应中获取文件输入流
InputStream is = conn.getInputStream();
byte[] b = new byte[1024];
int len = 0;
int total = 0;
//拿到临时文件的输出流
File file = new File("genymotion-2.2.0-vbox.exe");
//创建一个临时文件
RandomAccessFile raf = new RandomAccessFile(file, "rwd");
//把文件的写入位置移动至startIndex
raf.seek(startIndex);
while((len = is.read(b)) != -1) {
//每次读取流里数据之后,同步把数据写入临时文件
raf.write(b, 0, len);
total += len;
//System.out.println("线程" + threadId + "下载了" + total);
//生成一个专门用来记录下载进度的临时文件
RandomAccessFile progressRaf = new RandomAccessFile(progressFile, "rwd");
//每次读取流里数据之后,同步把当前线程下载的总进度写入进度临时文件中
progressRaf.write((total + "").getBytes());
//关闭进度临时文件
progressRaf.close();
}
System.out.println("线程" + threadId + "下载完毕-------------------小志参上!");
//关闭临时文件
raf.close();
//完成下载的线程的个数增加1
MultiDownload.finishedThread++;
//MultiDownload.path表示文件的下载地址
synchronized (MultiDownload.path) {
//如果三个下载线程都完成了下载任务
if(MultiDownload.finishedThread == MultiDownload.ThreadCount) {
//变量所有的下载线程所在的临时文件
for (int i = 0; i < MultiDownload.ThreadCount; i++) {
//获得下载线程下载的临时文件
File f = new File(i + ".txt");
//删除下载线程下载的临时文件
f.delete();
}
//设置结束下载任务的下载线程的个数为0
MultiDownload.finishedThread = 0;
}
}
}
} catch (Exception e) {
e.printStackTrace();
}
}
}
第三步:执行程序,等到Console输出框中输出下图中所示的信息表示下载完成
此时在Java工程下可以看到多了一个genymotion-2.2.0-vbox.exe,表示下载成功了
第一步:启动Tomcat服务器,将需要下载的文件部署到Tomcat服务器上
第二步:使用eclipse创建一个Java工程,并且在工程中添加下面的代码
package com.fyt.multidownload;
import java.io.BufferedReader;
import java.io.File;
import java.io.FileInputStream;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.io.RandomAccessFile;
import java.net.HttpURLConnection;
import java.net.URL;
public class MultiDownload {
//下载文件时开启的线程的个数
static int ThreadCount = 3;
//结束的线程的个数
static int finishedThread = 0;
//设置文件的下载地址
static String path = "http://192.168.0.101:8080/app/genymotion-2.2.0-vbox.exe";
public static void main(String[] args) {
try {
//将下载地址封装成Url对象
URL url = new URL(path);
//创建连接对象,此时未建立连接
HttpURLConnection conn = (HttpURLConnection) url.openConnection();
//设置请求方式为get请求
conn.setRequestMethod("GET");
//设置连接超时
conn.setConnectTimeout(5000);
//设置读取超时
conn.setReadTimeout(5000);
//如果请求发送成功
if(conn.getResponseCode() == 200) {
//获得需要下载的文件的长度
int length = conn.getContentLength();
//创建File对象
File file = new File("genymotion-2.2.0-vbox.exe");
//生成临时文件(临时文件的大小为下载的文件的大小)
RandomAccessFile raf = new RandomAccessFile(file, "rwd");
//设置临时文件的大小
raf.setLength(length);
//关闭临时文件
raf.close();
//计算出每个线程应该下载多少字节
int size = length / ThreadCount;
//遍历创建好的线程
for (int i = 0; i < ThreadCount; i++) {
//计算线程下载的开始位置
int startIndex = i * size;
//计算线程下载的结束位置
int endIndex = (i + 1) * size - 1;
//如果是最后一个线程,那么结束位置写死
if(i == ThreadCount - 1)
{
endIndex = length - 1;
}
//System.out.println("线程" + i + "的下载区间是:" + startIndex + "---" + endIndex);
//创建下载线程
//第一个参数:下载开始的位置
//第二个参数:下载结束的位置
//第三个参数:线程的Id
DownLoadThread thread = new DownLoadThread(startIndex, endIndex, i);
//启动下载线程
thread.start();
}
}
} catch (Exception e) {
e.printStackTrace();
}
}
}
//创建一个继承自线程类的下载线程类
class DownLoadThread extends Thread {
//线程下载的开始位置
int startIndex;
//线程下载的结束位置
int endIndex;
//线程的Id
int threadId;
//下载线程的构造方法
public DownLoadThread(int startIndex, int endIndex, int threadId) {
super();
//设置线程下载的开始位置
this.startIndex = startIndex;
//设置线程下载的结束位置
this.endIndex = endIndex;
//设置线程的Id
this.threadId = threadId;
}
//执行线程
@Override
public void run() {
try {
//创建保存下载进度的临时文件
File progressFile = new File(threadId + ".txt");
//如果进度临时文件存在
if(progressFile.exists()) {
//创建文件输入流
FileInputStream fis = new FileInputStream(progressFile);
//InputStreamReader:创建文件的输入流缓冲区对象
//BufferedReader:创建文件的读取缓冲区对象
BufferedReader br = new BufferedReader(new InputStreamReader(fis));
//从进度临时文件中读取出上一次下载的总进度
//然后与原本的开始位置相加,得到新的开始位置
startIndex += Integer.parseInt(br.readLine());
//关闭文件输入流
fis.close();
}
//System.out.println("线程" + threadId + "的下载区间是:" + startIndex + "---" + endIndex);
//使用URL对象封装下载文件的地址
URL url = new URL(MultiDownload.path);
//创建连接对象,此时未建立连接
HttpURLConnection conn = (HttpURLConnection) url.openConnection();
//设置请求方式为get请求
conn.setRequestMethod("GET");
//设置连接超时
conn.setConnectTimeout(5000);
//设置读取超时
conn.setReadTimeout(5000);
//设置本次http请求所请求的数据的区间
conn.setRequestProperty("Range", "bytes=" + startIndex + "-" + endIndex);
//请求部分数据
//请求部分数据请求成功的响应码是206
if(conn.getResponseCode() == 206) {
//流里此时只有1/3原文件的数据
//从响应中获取文件输入流
InputStream is = conn.getInputStream();
byte[] b = new byte[1024];
int len = 0;
int total = 0;
//拿到临时文件的输出流
File file = new File("genymotion-2.2.0-vbox.exe");
//创建一个临时文件
RandomAccessFile raf = new RandomAccessFile(file, "rwd");
//把文件的写入位置移动至startIndex
raf.seek(startIndex);
while((len = is.read(b)) != -1) {
//每次读取流里数据之后,同步把数据写入临时文件
raf.write(b, 0, len);
total += len;
//System.out.println("线程" + threadId + "下载了" + total);
//生成一个专门用来记录下载进度的临时文件
RandomAccessFile progressRaf = new RandomAccessFile(progressFile, "rwd");
//每次读取流里数据之后,同步把当前线程下载的总进度写入进度临时文件中
progressRaf.write((total + "").getBytes());
//关闭进度临时文件
progressRaf.close();
}
System.out.println("线程" + threadId + "下载完毕-------------------小志参上!");
//关闭临时文件
raf.close();
//完成下载的线程的个数增加1
MultiDownload.finishedThread++;
//MultiDownload.path表示文件的下载地址
synchronized (MultiDownload.path) {
//如果三个下载线程都完成了下载任务
if(MultiDownload.finishedThread == MultiDownload.ThreadCount) {
//变量所有的下载线程所在的临时文件
for (int i = 0; i < MultiDownload.ThreadCount; i++) {
//获得下载线程下载的临时文件
File f = new File(i + ".txt");
//删除下载线程下载的临时文件
f.delete();
}
//设置结束下载任务的下载线程的个数为0
MultiDownload.finishedThread = 0;
}
}
}
} catch (Exception e) {
e.printStackTrace();
}
}
}
第三步:执行程序,等到Console输出框中输出下图中所示的信息表示下载完成
此时在Java工程下可以看到多了一个genymotion-2.2.0-vbox.exe,表示下载成功了
相关文章推荐
- Eclipse中导入Heritrix,报错找不到类 sun.net.www.protocol.file.FileURLConnection
- java中静态块,静态方法,非静态方法,构造方法执行的顺序
- dbunit之org.dbunit.dataset.DataSetException: java.net.MalformedURLException解决方法
- Java程序如何生成Jar、exe及安装文件
- LeetCode144 Binary Tree Preorder Traversal(迭代实现) Java
- 通过字节码分析JDK8中Lambda表达式编译及执行机制
- Java千百问_05面向对象(011)_引用传递和值传递有什么区别
- JSTL(c标签)与Struts2(s标签)标签的常用功能对比
- SpringMVC -- 拦截器技术1 (6)
- Java的数据类型转换
- JAVA接口的简单实现
- Java九九乘法表/冒泡排序倒叙输出
- The Java™ Tutorials
- spring多数据源配置
- java中的集合类
- 《day16_多线程细节_Eclipse使用》
- eclipse中 将java项目转换为web项目
- Java web:SSH项目拷贝产生问题以及解决(getContextPath() xml hibernate等报错)
- Myeclipse10.x破解方法
- java基础算法学习之分治