大家好,我是你们的志哥。
今天打算分享一下多线程下载的原理。
假如我们把一个服务器上的文件看作是一个水缸里的水的话,那么多线程下载就相当于从水缸上打了多个小孔,然后塞进去小管道进行抽水。呵呵,也许这个比喻不够准确。多线程下载大致可分为以下几个步骤:
一、首先在本地创建一个与服务器文件大小相同的临时文件(这个很好理解,如果我想下个2G的电影,我得给先在本地占用2G的空间,不然不能下着下着没空间了是吧)。
二、计算分配几个线程去下载服务器上的资源,知道每个线程下载文件的起始位置。
那么这个起始位置怎么计算呢?
文件长度/线程个数= 每个线程下载文件的大小。那么
线程1下载的位置:0~每个线程下载文件的大小-1.
线程2:以此类推
那么就是i线程的下载起始位置: (i-1)*每个线程下载文件的大小
三、开启多个线程,每一个线程下载对应位置的文件。
四、如果所有的线程都把自己的数据下载完毕了,服务器上的资源就被下载到本地了。
五、当文件都下载到本地了,那么还有一个文件就是把各个线程下载的文件如何串起来。那么就要利用到一个类:RandomAccessFile 随机文件访问类。
代码如下:
- import java.io.InputStream;
- import java.io.RandomAccessFile;
- import java.net.HttpURLConnection;
- import java.net.URL;
-
-
- public class Demo {
- public static int threadCount = 3;
-
-
-
- public static void main(String[] args) throws Exception{
-
- String path ="http://192.168.1.100:8080/360.exe";
- URL url = new URL(path);
- HttpURLConnection conn = (HttpURLConnection) url.openConnection();
- conn.setConnectTimeout(5000);
- conn.setRequestMethod("GET");
- int code = conn.getResponseCode();
- if(code==200){
-
- int length = conn.getContentLength();
- System.out.println("文件总长度:"+length);
- RandomAccessFile raf = new RandomAccessFile("setup.exe", "rwd");
-
- raf.setLength(length);
- raf.close();
-
-
-
- int blockSize = length / threadCount;
-
- for(int threadId=1;threadId<=threadCount;++threadId){
-
- int startIndex = (threadId-1)*blockSize;
- int endIndex = blockSize - 1;
- if(threadId==threadCount){
-
- endIndex = length;
- }
- System.out.println("线程:"+threadId+"下载:--"+startIndex+"-->"+endIndex);
- new DownLoadThread(threadId, startIndex, endIndex, path).start();
- }
- }else{
- System.out.println("访问错误");
- }
-
- }
-
-
-
-
-
- public static class DownLoadThread extends Thread{
- private int threadId;
- private int startIndex;
- private int endIndex;
- private String path;
-
-
-
-
-
-
- public DownLoadThread(int threadId, int startIndex, int endIndex,String path) {
- this.threadId = threadId;
- this.startIndex = startIndex;
- this.endIndex = endIndex;
- this.path = path;
- }
-
- @Override
- public void run() {
- try{
- URL url = new URL(path);
- HttpURLConnection conn = (HttpURLConnection) url.openConnection();
- conn.setRequestMethod("GET");
-
- conn.setRequestProperty("Range", "bytes="+startIndex+"-"+endIndex);
- conn.setConnectTimeout(5000);
- int code = conn.getResponseCode();
- System.out.println("code="+code);
-
- InputStream is = conn.getInputStream();
- RandomAccessFile raf = new RandomAccessFile("setup.exe", "rwd");
-
- raf.seek(startIndex);
-
- int len =0;
- byte[] buffer = new byte[1024];
- while((len = is.read(buffer)) != -1){
- raf.write(buffer,0,len);
- }
- is.close();
- raf.close();
- System.out.println("线程"+threadId+"下载完毕");
-
- }catch(Exception e){
- e.printStackTrace();
- }
- }
-
- }
-
- }