您的位置:首页 > 理论基础 > 计算机网络

HttpURLConnection实现多线程下载

2016-11-06 09:41 197 查看

HttpURLConnection下载

HttpURLConnection下载步骤

1,将要下载的路径封装成一个url对象(一定是可以下载的路径)
2,通过url获取conn,并设置conn的各种属性,最重要的是setRequestMethod("GET");
3,判断code,如果是200,通过conn获取要下载的文件大小len
4,在本地创建一个RandomAccessFile文件,并设置长度 raf.setLength(len);
5,定义线程个数threadCount,每个线程要下载的字节大小threadSize,线程的开始下载位置startIndex,结束下载位置endIndex
6,for循环,给线程的threadId,线程的开始位置startIndex,endIndex进行赋值,并开启线程
7,定义一个MyThread类继承Thread,要放在主类的外面
8,定义threadId,startIndex,endIndex,并通过构造方法进行初始化
9,将要下载的路径封装成url对象
10,设置conn的属性,是重要的是设置请求头属性range ,conn.setRequestProperty("range" , "bytes="+startIndex+"-"+endIndex);
如果range属性设置错误会出现416,作用,设置请求的文件的起始位置
11,获取响应码,如果是206,获取输入流
12,创建一个RandomAccessFile,并设置raf.seek(startIndex),表示从这个位置开始写出
13,将输入流is中的数据放入到内存,raf将内存中的数据写出到文件,并可以设置写出的位置
14,关闭输入流is和随机访问流raf

注意事项:
1,要下载的路径一定是一个可下载的文件
2,在知道要下载文件大小后,通过raf创建本地文件,一定要设置文件的大小为len
3,在子线程中设置请求头range的时候一定要注意格式,conn.setRequestProperty("range","bytes:"+startIndex+"-"+endIndex);不然会出现416错误
4,在创建raf对象后,一定要设置要写的开始位置 raf.seek(startIndex);
5,下载的时候是请求方式是get,一定不能是Post不然会是411错误


HttpURLConnection断点续传下载步骤

1,在循环往外的写前,定义一个total用来记录从开始到现在写的字节个数
2,在循环里面,定义一个线程的当前位置int currentIndex = total + startIndex;
3,创建一个随机访问流,并设置模式为"rwd",直接写入硬盘
4,在设置请求头range前,创建一个bufferedReader,读取文件里面的数据,并将数据转换为int类型
5,然后将startIndex = 读取的结果

注意事项:
要使用RandomAccessFile(file,"rwd")来写,不然写出的文件内容为0,这样读取的时候字符串转换为数据就为异常


HttpURLConnection断点续传完善

问题,每次下载完,.txt文件也写完了,如果不删除,下次下载的时候直接下载完成了
解决方案:
1,在每个线程下载完之后,就将该.txt文件删除
2,删除如果删不掉,因为没有写完之后,没有关raf的流


package com.heima.download;

import java.io.BufferedReader;
import java.io.BufferedWriter;
import java.io.File;
import java.io.FileOutputStream;
import java.io.FileReader;
import java.io.FileWriter;
import java.io.InputStream;
import java.io.RandomAccessFile;
import java.net.HttpURLConnection;
import java.net.MalformedURLException;
import java.net.URL;

public class Download {
public static void main(String[] args) {
/*
*
* 从服务器上下载的步骤,上传需要使用AsncHttpClient 但是下载,的话不需要 下载步骤 1,将请求的路径封装成一个Url路径对象
* 2,通过url获取流is 3,通过is获取文件的大小
* 4,在本地使用RandomAcessFile创建一个大文件,并设置大小跟服务器上一样大
* 5,定义线程个数threadCount,每个线程要下载的大小threadSize,线程的起始位置startIndex 线程的结束位置
* endIndex 6,使用for循环初始化线程的id,threadId,startIndex,endIndex,创建线程并开启
* 线程的结束位置,如果是最后一个线程endIndex = len - 1;
*
* 7,在类中创建一个内部类MyThread继承Thread 8,创建一个构造方法,初始化threadId,startIndex
* ,endIndex 9,将要下载的请求路径封装成Url对象,并获取流对象
* 10,设置流的conn.setRequestProperty("Range"
* ,"bytes="+startIndex+"-"+endIndex); 11,判断响应码,如果是206的话
* 12,将读入到的流使用RandomAccessFile raf进行写出
*/

try {
// 1,将请求的下载路径封装成一个url对象
//          URL url = new URL("http://localhost:8080/itheima74/feiq.exe");
URL url = new URL("http://img1.3lian.com/2015/w2/60/d/41.jpg");
// 获取httpurlconnection连接
HttpURLConnection conn = (HttpURLConnection) url.openConnection();
// 设置连接属性
conn.setRequestMethod("GET");
conn.setReadTimeout(5000);
conn.setConnectTimeout(5000);
int code = conn.getResponseCode();
System.out.println(code);
if (code == 200) {
// 通过连接获取要下载的文件大小
int len = conn.getContentLength();
// 在本地创建一个大小一样的RandomAccessFile文件,格式也要一样
//              RandomAccessFile raf = new RandomAccessFile(
//                      new File("fuck.exe"), "rw");
RandomAccessFile raf = new RandomAccessFile(
new File("fuck.jpg"), "rw");
//一定要文件的大小
raf.setLength(len);
// 定义线程的个数threadCount,每个线程的要下载的大小 threadSize,线程的起始位置startIndex
// 结束位置endIndex
int threadCount = 3;
int startIndex;
int endIndex;
int threadSize = len / threadCount;
for (int threadId = 0; threadId < threadCount; threadId++) {
startIndex = threadId * threadSize;
endIndex = (threadId + 1) * threadSize - 1;

4000
if (threadId == threadCount - 1) {
endIndex = len - 1;
}
System.out.println(threadId);
new MyThread(threadId, startIndex, endIndex).start();
}
}

} catch (Exception e) {

}

}

}

class MyThread extends Thread {
int threadId;
int startIndex;
int endIndex;

// 通过构造方法给线程的id,线程的开始位置,线程的结束位置进行初始化
MyThread(int threadId, int startIndex, int endIndex) {
this.threadId = threadId;
this.startIndex = startIndex;
this.endIndex = endIndex;
}

@Override
public void run() {
// 将要下载的路径封装成一个URL对象
try {
//          URL url = new URL("http://localhost:8080/itheima74/feiq.exe");
URL url = new URL("http://img1.3lian.com/2015/w2/60/d/41.jpg");
// 通过url获取连接对象
HttpURLConnection conn = (HttpURLConnection) url.openConnection();
// 设置连接对象的属性
conn.setRequestMethod("GET");
conn.setConnectTimeout(5000);
conn.setReadTimeout(5000);
File file = new File(threadId+".txt");
if(file.exists()){
BufferedReader br = new BufferedReader(new FileReader(file));
startIndex = Integer.parseInt(br.readLine());
br.close();
}
System.out.println("线程"+threadId+"从"+startIndex+"开始下载");
//一定要记住要从某个位置到另一个位置下载,一定要设置请求头Range
conn.setRequestProperty("Range", "bytes="+startIndex+"-"+endIndex);
int code = conn.getResponseCode();
//部分请求的响应码为206
if(code == 206){

//获取响应的字节输入流
System.out.println("线程"+threadId+"开始下载");
InputStream is = conn.getInputStream();
//              RandomAccessFile raf = new RandomAccessFile(new File("fuck.exe"), "rw");
RandomAccessFile raf = new RandomAccessFile(new File("fuck.jpg"), "rw");
//              设置随机访问流开始写的位置
raf.seek(startIndex);
int len = -1;   //有效的字节码长度
byte[] buffer = new byte[1024*10];//定义一个小数组进行读入
int total = 0;
while((len = is.read(buffer)) != -1){
raf.write(buffer, 0, len);

total = total + len;
int currentIndex = total + startIndex;

//将当前读入的字节数写出到本地文件
RandomAccessFile raf1 = new RandomAccessFile(new File(threadId+".txt"), "rwd");
raf1.write((currentIndex+"").getBytes());
raf1.close();

}

//关闭读入和字节流和写出的随机文件访问流
is.close();
raf.close();

System.out.println("线程"+threadId+"下载结束");
File file1 = new File(threadId+".txt");
file1.delete();

}

} catch (Exception e) {
// TODO Auto-generated catch block
e.printStackTrace();
}

}
}
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息