多线程下载文件的思路:
1.首先获取到文件的总大小
获取文件大小的方式是通过网络读取,getContentLength()即可获取到文件的大小,使用RandomAccessFile()支持随机访问
2.根据所准备的线程数据,计算每一个线程需要下载的文件的大小
上图显示下载400M的电影分4个线程下载,每一个线程分别下载各自数据段中的数据,第一个线程下载0-100M,第二个下载100M-200M之间的数据,依次类推。因此下载过程中需要记住的是的开始位置段和结束位置段,其实只需要开始位置就可以了,结束为止可以根据开始位置加上下载的大小来推断获取。
3.获取到大小数据以后,开始用线程循环读取每一个区间的数据
这个里面需要注意的是,要更新数据的写入位置seek(startIndex),逐段填满,不然会出现覆盖以前的数据。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
|
package com.ldw.multilthreaddownload;
import java.io.File;
import java.io.InputStream;
import java.io.RandomAccessFile;
import java.net.HttpURLConnection;
import java.net.URL;
public class Multidownload {
static int ThreadCount = 3 ; //线程的个数
public static void main(String[] args) {
// TODO Auto-generated method stub
//发送get请求,请求这个地址的资源
try {
URL url = new URL(path);
HttpURLConnection conn = (HttpURLConnection) url.openConnection();
conn.setRequestMethod( "GET" );
conn.setConnectTimeout( 5000 );
conn.setReadTimeout( 5000 );
if (conn.getResponseCode() == 200 ){
//获取到请求资源文件的长度
int length = conn.getContentLength();
File file = new File( "QQ.exe" );
//创建随机存储文件
RandomAccessFile raf = new RandomAccessFile(file, "rwd" );
//设置临时文件的大小
raf.setLength(length);
//关闭raf
raf.close();
//计算出每一个线程下载多少字节
int size = length / Multidownload.ThreadCount;
for ( int i = 0 ; i < Multidownload.ThreadCount; i ++){
//startIndex,endIndex分别代表线程的开始和结束位置
int startIndex = i * size;
int endIndex = (i + 1 ) * size - 1 ;
if (i == ThreadCount - 1 ){
//如果是最后一个线程,那么结束位置写死
endIndex = length - 1 ;
}
System.out.println( "线程" + i + "的下载区间是" + startIndex + "到" + endIndex);
new DownLoadThread(startIndex, endIndex, i).start(); //创建线程下载数据
}
}
} catch (Exception e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
}
class DownLoadThread extends Thread{
int startIndex;
int endIndex;
int threadId;
public DownLoadThread( int startIndex, int endIndex, int threadId) {
super ();
this .startIndex = startIndex;
this .endIndex = endIndex;
this .threadId = threadId;
}
@Override
public void run(){
//使用http请求下载安装包文件
URL url;
try {
url = new URL(Multidownload.path);
HttpURLConnection conn = (HttpURLConnection) url.openConnection();
conn.setRequestMethod( "GET" );
conn.setConnectTimeout( 5000 );
conn.setReadTimeout( 5000 );
//设置请求数据的区间
conn.setRequestProperty( "Range" , "bytes=" + startIndex + "-" + endIndex);
//请求部分数据的响应码是206
if (conn.getResponseCode() == 206 ){
//获取一部分数据来读取
InputStream is = conn.getInputStream();
byte [] b = new byte [ 1024 ];
int len = 0 ;
int total = 0 ;
//拿到临时文件的引用
File file = new File( "QQ.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);
}
System.out.println( "线程" + threadId + "下载过程结束===========================" );
raf.close();
}
} catch (Exception e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
};
}
|
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持服务器之家。