Java多线程下载

本文介绍了一种使用多线程技术实现文件分段下载的方法。通过将文件分为多个部分并为每一部分分配一个线程进行下载,可以显著提高下载速度。文章提供了完整的Java代码示例,展示了如何创建线程池、计算每个线程负责的数据量以及如何利用HTTP Range请求头来下载文件的特定部分。
通过多线程对文件分段进行下载,废话不多说,直接上代码
package com.download.test;

 
import java.io.File;
import java.net.HttpURLConnection;
import java.net.URL;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
 
/*
 * 多线程下载
 */
public class DownloadTest {
    //初始一个带缓存的线程池
    public static ExecutorService executorService = Executors.newCachedThreadPool();
    
	public static void main(String[] args) throws Exception {
        String path = "http://csdn-app.youkuaiyun.com/csdn.apk";
        new DownloadTest().download(path,3);
    }
     
    public void download (String path,int threadsize) throws Exception{
        URL url = new URL(path);
        HttpURLConnection conn = (HttpURLConnection) url.openConnection();
        conn.setConnectTimeout(3000);
        conn.setRequestMethod("GET");
        if(conn.getResponseCode() == 200){
            //获取网络文件长度,文件的大小(单位是字节)
            int length = conn.getContentLength();   
            //新建本地文件保存下载数据
            File file = new File("D:/"+getFilename(path));
            //计算每条线程负责下载的数据量
            int block = length%threadsize==0 ? length/threadsize : length/threadsize+1;
            //开启指定数目的线程同时下载
            for(int threadid = 0; threadid < threadsize; threadid++){
            	executorService.execute(new DownloadThread(threadid,block,url,file,length));
                //new DownloadThread(threadid,block,url,file,length).start();
            }
        }else{
            System.out.println("下载失败!");
        }
    }
     
    private String getFilename(String path) {
        return path.substring(path.lastIndexOf("/")+1);
    }
}


下载线程类

package com.download.test;

import java.io.File;
import java.io.InputStream;
import java.io.RandomAccessFile;
import java.net.HttpURLConnection;
import java.net.URL;

public class DownloadThread extends Thread{
	private static int finished;
    private int threadid;   //线程编号
    private int block;      //下载块大小
    private URL url;        //下载链接
    private File file;      //下载数据保存文件
    private int length;		//文件长度
    
    public DownloadThread(int threadid, int block, URL url, File file,int length) {
        this.threadid = threadid;
        this.block = block;
        this.url = url;
        this.file = file;
        this.length = length;
    }

    public void run() {
        int start = threadid * block;       //本线程下载数据写入文件开始位置
        int end = (threadid+1) * block - 1; //本线程下载数据写入文件结束位置
        try {
            //创建一个随机访问文件流对象
            RandomAccessFile accessFile = new RandomAccessFile(file, "rwd");
            //文件指针偏移至正确写入位置
            accessFile.seek(start);
            HttpURLConnection conn = (HttpURLConnection) url.openConnection();
            conn.setConnectTimeout(3000);
            conn.setRequestMethod("GET");
            //设置请求数据的范围
            conn.setRequestProperty("Range", "bytes="+start+"-"+end);
            if(conn.getResponseCode() == 206){//状态码206:(部分内容) 服务器成功处理了部分 GET 请求
                InputStream inStream = conn.getInputStream();
                long time = System.currentTimeMillis();
                byte[] buffer = new byte[1024];
                int len = 0;
                while((len = inStream.read(buffer)) != -1){
                    accessFile.write(buffer, 0, len);
                    finished += len;
                    int i = finished * 100 / length;
                    //System.out.println(finished+"----------"+length);
                    if(System.currentTimeMillis()-time>500 || finished>=length){
                    	time = System.currentTimeMillis();
                    	System.out.println("下载进度:"+i+"%");
                    }
                }
                accessFile.close();
                inStream.close();
            }
            System.out.println("第"+(threadid+1)+"部分下载完成");
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
}



1.得到服务器下载文件的大小,然后在本地设置一个临时文件和服务器端文件大小一致 a)获得访问网络地址 b)通过URL对象的openConnection()方法打开连接,返回一个连接对象 c)设置请求头 i.setRequestMethod ii.setConnectTimeout iii.setReadTimeout d)判断是否响应成功 e)获取文件长度(getContentLength()) f)随机访问文件的读取与写入RandomAccessFile(file, mode) g)设置临时文件与服务器文件大小一致(setLength()) h)关闭临时文件 2.计算出每个线程下载的大小(开始位置,结束位置) a)计算出每个线程下载的大小 b)for循环,计算出每个线程的开始、结束位置 c)最后一个线程处理 3.每创建好一次就要开启线程下载 a)构造方法 b)通过URL对象的openConnection()方法打开连接,返回一个连接对象 c)设置请求头 i.setRequestMethod ii.setConnectTimeout d)判断是否响应成功(206) e)获取每个线程返回的流对象 f)随机访问文件的读取与写入RandomAccessFile(file, mode) g)指定开始位置 h)循环读取 i.保存每个线程下载位置 ii.记录每次下载位置 iii.关闭临时记录位置文件 iv.随机本地文件写入 v.记录已下载大小 i)关闭临时文件 j)关闭输入流 4.为了杀死线程还能继续下载的情况下,从本地文件上读取已经下载文件的开始位置 a)创建保存记录结束位置的文件 b)读取文件 c)将流转换为字符 d)获取记录位置 e)把记录位置赋给开始位置 5.当你的n个线程都下载完毕的时候我进行删除记录下载位置的缓存文件 a)线程下载完就减去 b)当没有正在运行的线程时切文件存在时删除文件
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值