zoukankan      html  css  js  c++  java
  • Java 多线程断点下载文件

    基本原理:利用URLConnection获取要下载文件的长度、头部等相关信息,并设置响应的头部信息。并且通过URLConnection获取输入流,将文件分成指定的块,每一块单独开辟一个线程完成数据的读取、写入。通过输入流读取下载文件的信息,然后将读取的信息用RandomAccessFile随机写入到本地文件中。同时,每个线程写入的数据都文件指针也就是写入数据的长度,需要保存在一个临时文件中。这样当本次下载没有完成的时候,下次下载的时候就从这个文件中读取上一次下载的文件长度,然后继续接着上一次的位置开始下载。并且将本次下载的长度写入到这个文件中。

    个人博客:

    http://hoojo.cnblogs.com

    http://blog.csdn.net/IBM_hoojo

    email: hoojo_@126.com

     

    一、下载文件信息类、实体

    封装即将下载资源的信息

    package com.hoo.entity;
     
    /**
     * <b>function:</b> 下载文件信息类
     * @author hoojo
     * @createDate 2011-9-21 下午05:14:58
     * @file DownloadInfo.java
     * @package com.hoo.entity
     * @project MultiThreadDownLoad
     * @blog http://blog.csdn.net/IBM_hoojo
     * @email hoojo_@126.com
     * @version 1.0
     */
    public class DownloadInfo {
        //下载文件url
        private String url;
        //下载文件名称
        private String fileName;
        //下载文件路径
        private String filePath;
        //分成多少段下载, 每一段用一个线程完成下载
        private int splitter;
        
        //下载文件默认保存路径
        private final static String FILE_PATH = "C:/temp";
        //默认分块数、线程数
        private final static int SPLITTER_NUM = 5;
        
        public DownloadInfo() {
            super();
        }
        
        /**
         * @param url 下载地址
         */
        public DownloadInfo(String url) {
            this(url, null, null, SPLITTER_NUM);
        }
        
        /**
         * @param url 下载地址url
         * @param splitter 分成多少段或是多少个线程下载
         */
        public DownloadInfo(String url, int splitter) {
            this(url, null, null, splitter);
        }
        
        /***
         * @param url 下载地址
         * @param fileName 文件名称
         * @param filePath 文件保存路径
         * @param splitter 分成多少段或是多少个线程下载
         */
        public DownloadInfo(String url, String fileName, String filePath, int splitter) {
            super();
            if (url == null || "".equals(url)) {
                throw new RuntimeException("url is not null!");
            }
            this.url =  url;
            this.fileName = (fileName == null || "".equals(fileName)) ? getFileName(url) : fileName;
            this.filePath = (filePath == null || "".equals(filePath)) ? FILE_PATH : filePath;
            this.splitter = (splitter < 1) ? SPLITTER_NUM : splitter;
        }
        
        /**
         * <b>function:</b> 通过url获得文件名称
         * @author hoojo
         * @createDate 2011-9-30 下午05:00:00
         * @param url
         * @return
         */
        private String getFileName(String url) {
            return url.substring(url.lastIndexOf("/") + 1, url.length());
        }
        
        public String getUrl() {
            return url;
        }
     
        public void setUrl(String url) {
            if (url == null || "".equals(url)) {
                throw new RuntimeException("url is not null!");
            }
            this.url = url;
        }
     
        public String getFileName() {
            return fileName;
        }
     
        public void setFileName(String fileName) {
            this.fileName = (fileName == null || "".equals(fileName)) ? getFileName(url) : fileName;
        }
     
        public String getFilePath() {
            return filePath;
        }
     
        public void setFilePath(String filePath) {
            this.filePath = (filePath == null || "".equals(filePath)) ? FILE_PATH : filePath;
        }
     
        public int getSplitter() {
            return splitter;
        }
     
        public void setSplitter(int splitter) {
            this.splitter = (splitter < 1) ? SPLITTER_NUM : splitter;
        }
        
        @Override
        public String toString() {
            return this.url + "#" + this.fileName + "#" + this.filePath + "#" + this.splitter;
        }
    }
     

     

    二、随机写入一段文件

    package com.hoo.download;
     
    import java.io.IOException;
    import java.io.RandomAccessFile;
     
    /**
     * <b>function:</b> 写入文件、保存文件
     * @author hoojo
     * @createDate 2011-9-21 下午05:44:02
     * @file SaveItemFile.java
     * @package com.hoo.download
     * @project MultiThreadDownLoad
     * @blog http://blog.csdn.net/IBM_hoojo
     * @email hoojo_@126.com
     * @version 1.0
     */
    public class SaveItemFile {
        //存储文件
        private RandomAccessFile itemFile;
        
        public SaveItemFile() throws IOException {
            this("", 0);
        }
        
        /**
         * @param name 文件路径、名称
         * @param pos 写入点位置 position
         * @throws IOException
         */
        public SaveItemFile(String name, long pos) throws IOException {
            itemFile = new RandomAccessFile(name, "rw");
            //在指定的pos位置开始写入数据
            itemFile.seek(pos);
        }
        
        /**
         * <b>function:</b> 同步方法写入文件
         * @author hoojo
         * @createDate 2011-9-26 下午12:21:22
         * @param buff 缓冲数组
         * @param start 起始位置
         * @param length 长度
         * @return
         */
        public synchronized int write(byte[] buff, int start, int length) {
            int i = -1;
            try {
                itemFile.write(buff, start, length);
                i = length;
            } catch (IOException e) {
                e.printStackTrace();
            }
            return i;
        }
        
        public void close() throws IOException {
            if (itemFile != null) {
                itemFile.close();
            }
        }
    }
    这个类主要是完成向本地的指定文件指针出开始写入文件,并返回当前写入文件的长度(文件指针)。这个类将被线程调用,文件被分成对应的块后,将被线程调用。每个线程都将会调用这个类完成文件的随机写入。

    三、单个线程下载文件

    package com.hoo.download;
     
    import java.io.IOException;
    import java.io.InputStream;
    import java.net.HttpURLConnection;
    import java.net.MalformedURLException;
    import java.net.URL;
    import java.net.URLConnection;
    import com.hoo.util.LogUtils;
     
    /**
     * <b>function:</b> 单线程下载文件
     * @author hoojo
     * @createDate 2011-9-22 下午02:55:10
     * @file DownloadFile.java
     * @package com.hoo.download
     * @project MultiThreadDownLoad
     * @blog http://blog.csdn.net/IBM_hoojo
     * @email hoojo_@126.com
     * @version 1.0
     */
    public class DownloadFile extends Thread {
        
        //下载文件url
        private String url;
        //下载文件起始位置  
        private long startPos;
        //下载文件结束位置
        private long endPos;
        //线程id
        private int threadId;
        
        //下载是否完成
        private boolean isDownloadOver = false;
     
        private SaveItemFile itemFile;
        
        private static final int BUFF_LENGTH = 1024 * 8;
        
        /**
         * @param url 下载文件url
         * @param name 文件名称
         * @param startPos 下载文件起点
         * @param endPos 下载文件结束点
         * @param threadId 线程id
         * @throws IOException
         */
        public DownloadFile(String url, String name, long startPos, long endPos, int threadId) throws IOException {
            super();
            this.url = url;
            this.startPos = startPos;
            this.endPos = endPos;
            this.threadId = threadId;
            //分块下载写入文件内容
            this.itemFile = new SaveItemFile(name, startPos);
        }
     
        
        @Override
        public void run() {
            while (endPos > startPos && !isDownloadOver) {
                try {
                    URL url = new URL(this.url);
                    HttpURLConnection conn = (HttpURLConnection) url.openConnection();
                    
                    // 设置连接超时时间为10000ms
                    conn.setConnectTimeout(10000);
                    // 设置读取数据超时时间为10000ms
                    conn.setReadTimeout(10000);
                    
                    setHeader(conn);
                    
                    String property = "bytes=" + startPos + "-";
                    conn.setRequestProperty("RANGE", property);
                    
                    //输出log信息
                    LogUtils.log("开始 " + threadId + ":" + property + endPos);
                    //printHeader(conn);
                    
                    //获取文件输入流,读取文件内容
                    InputStream is = conn.getInputStream();
                    
                    byte[] buff = new byte[BUFF_LENGTH];
                    int length = -1;
                    LogUtils.log("#start#Thread: " + threadId + ", startPos: " + startPos + ", endPos: " + endPos);
                    while ((length = is.read(buff)) > 0 && startPos < endPos && !isDownloadOver) {
                        //写入文件内容,返回最后写入的长度
                        startPos += itemFile.write(buff, 0, length);
                    }
                    LogUtils.log("#over#Thread: " + threadId + ", startPos: " + startPos + ", endPos: " + endPos);
                    LogUtils.log("Thread " + threadId + " is execute over!");
                    this.isDownloadOver = true;
                } catch (MalformedURLException e) {
                    e.printStackTrace();
                } catch (IOException e) {
                    e.printStackTrace();
                } finally {
                    try {
                        if (itemFile != null) {
                            itemFile.close();
                        }
                    } catch (IOException e) {
                        e.printStackTrace();
                    }
                }
            }
            if (endPos < startPos && !isDownloadOver) {
                LogUtils.log("Thread " + threadId  + " startPos > endPos, not need download file !");
                this.isDownloadOver = true;
            }
            if (endPos == startPos && !isDownloadOver) {
                LogUtils.log("Thread " + threadId  + " startPos = endPos, not need download file !");
                this.isDownloadOver = true;
            }
        }
        
        /**
         * <b>function:</b> 打印下载文件头部信息
         * @author hoojo
         * @createDate 2011-9-22 下午05:44:35
         * @param conn HttpURLConnection
         */
        public static void printHeader(URLConnection conn) {
            int i = 1;
            while (true) {
                String header = conn.getHeaderFieldKey(i);
                i++;
                if (header != null) {
                    LogUtils.info(header + ":" + conn.getHeaderField(i));
                } else {
                    break;
                }
            }
        }
        
        /**
         * <b>function:</b> 设置URLConnection的头部信息,伪装请求信息
         * @author hoojo
         * @createDate 2011-9-28 下午05:29:43
         * @param con
         */
        public static void setHeader(URLConnection conn) {
            conn.setRequestProperty("User-Agent", "Mozilla/5.0 (X11; U; Linux i686; en-US; rv:1.9.0.3) Gecko/2008092510 Ubuntu/8.04 (hardy) Firefox/3.0.3");
            conn.setRequestProperty("Accept-Language", "en-us,en;q=0.7,zh-cn;q=0.3");
            conn.setRequestProperty("Accept-Encoding", "utf-8");
            conn.setRequestProperty("Accept-Charset", "ISO-8859-1,utf-8;q=0.7,*;q=0.7");
            conn.setRequestProperty("Keep-Alive", "300");
            conn.setRequestProperty("connnection", "keep-alive");
            conn.setRequestProperty("If-Modified-Since", "Fri, 02 Jan 2009 17:00:05 GMT");
            conn.setRequestProperty("If-None-Match", "\"1261d8-4290-df64d224\"");
            conn.setRequestProperty("Cache-conntrol", "max-age=0");
            conn.setRequestProperty("Referer", "http://www.baidu.com");
        }
        
        public boolean isDownloadOver() {
            return isDownloadOver;
        }
        
        public long getStartPos() {
            return startPos;
        }
     
        public long getEndPos() {
            return endPos;
        }
    }

    这个类主要是完成单个线程的文件下载,将通过URLConnection读取指定url的资源信息。然后用InputStream读取文件内容,然后调用调用SaveItemFile类,向本地写入当前要读取的块的内容。

     

    四、分段多线程写入文件内容

    package com.hoo.download;
     
    import java.io.DataInputStream;
    import java.io.DataOutputStream;
    import java.io.File;
    import java.io.FileInputStream;
    import java.io.FileOutputStream;
    import java.io.IOException;
    import java.net.HttpURLConnection;
    import java.net.MalformedURLException;
    import java.net.URL;
    import com.hoo.entity.DownloadInfo;
    import com.hoo.util.LogUtils;
     
    /**
     * <b>function:</b> 分批量下载文件
     * @author hoojo
     * @createDate 2011-9-22 下午05:51:54
     * @file BatchDownloadFile.java
     * @package com.hoo.download
     * @project MultiThreadDownLoad
     * @blog http://blog.csdn.net/IBM_hoojo
     * @email hoojo_@126.com
     * @version 1.0
     */
    public class BatchDownloadFile implements Runnable {
        //下载文件信息 
        private DownloadInfo downloadInfo;
        //一组开始下载位置
        private long[] startPos;
        //一组结束下载位置
        private long[] endPos;
        //休眠时间
        private static final int SLEEP_SECONDS = 500;
        //子线程下载
        private DownloadFile[] fileItem;
        //文件长度
        private int length;
        //是否第一个文件
        private boolean first = true;
        //是否停止下载
        private boolean stop = false;
        //临时文件信息
        private File tempFile;
        
        public BatchDownloadFile(DownloadInfo downloadInfo) {
            this.downloadInfo = downloadInfo;
            String tempPath = this.downloadInfo.getFilePath() + File.separator + downloadInfo.getFileName() + ".position";
            tempFile = new File(tempPath);
            //如果存在读入点位置的文件
            if (tempFile.exists()) {
                first = false;
                //就直接读取内容
                try {
                    readPosInfo();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            } else {
                //数组的长度就要分成多少段的数量
                startPos = new long[downloadInfo.getSplitter()];
                endPos = new long[downloadInfo.getSplitter()];
            }
        }
        
        @Override
        public void run() {
            //首次下载,获取下载文件长度
            if (first) {
                length = this.getFileSize();//获取文件长度
                if (length == -1) {
                    LogUtils.log("file length is know!");
                    stop = true;
                } else if (length == -2) {
                    LogUtils.log("read file length is error!");
                    stop = true;
                } else if (length > 0) {
                    /**
                     * eg 
                     * start: 1, 3, 5, 7, 9
                     * end: 3, 5, 7, 9, length
                     */
                    for (int i = 0, len = startPos.length; i < len; i++) {
                        int size = i * (length / len);
                        startPos[i] = size;
                        
                        //设置最后一个结束点的位置
                        if (i == len - 1) {
                            endPos[i] = length;
                        } else {
                            size = (i + 1) * (length / len);
                            endPos[i] = size;
                        }
                        LogUtils.log("start-end Position[" + i + "]: " + startPos[i] + "-" + endPos[i]);
                    }
                } else {
                    LogUtils.log("get file length is error, download is stop!");
                    stop = true;
                }
            }
            
            //子线程开始下载
            if (!stop) {
                //创建单线程下载对象数组
                fileItem = new DownloadFile[startPos.length];//startPos.length = downloadInfo.getSplitter()
                for (int i = 0; i < startPos.length; i++) {
                    try {
                        //创建指定个数单线程下载对象,每个线程独立完成指定块内容的下载
                        fileItem[i] = new DownloadFile(
                            downloadInfo.getUrl(), 
                            this.downloadInfo.getFilePath() + File.separator + downloadInfo.getFileName(), 
                            startPos[i], endPos[i], i
                        );
                        fileItem[i].start();//启动线程,开始下载
                        LogUtils.log("Thread: " + i + ", startPos: " + startPos[i] + ", endPos: " + endPos[i]);
                    } catch (IOException e) {
                        e.printStackTrace();
                    }
                }
                
                //循环写入下载文件长度信息
                while (!stop) {
                    try {
                        writePosInfo();
                        LogUtils.log("downloading……");
                        Thread.sleep(SLEEP_SECONDS);
                        stop = true;
                    } catch (IOException e) {
                        e.printStackTrace();
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                    for (int i = 0; i < startPos.length; i++) {
                        if (!fileItem[i].isDownloadOver()) {
                            stop = false;
                            break;
                        }
                    }
                }
                LogUtils.info("Download task is finished!");
            }
        }
        
        /**
         * 将写入点数据保存在临时文件中
         * @author hoojo
         * @createDate 2011-9-23 下午05:25:37
         * @throws IOException
         */
        private void writePosInfo() throws IOException {
            DataOutputStream dos = new DataOutputStream(new FileOutputStream(tempFile));
            dos.writeInt(startPos.length);
            for (int i = 0; i < startPos.length; i++) {
                dos.writeLong(fileItem[i].getStartPos());
                dos.writeLong(fileItem[i].getEndPos());
                //LogUtils.info("[" + fileItem[i].getStartPos() + "#" + fileItem[i].getEndPos() + "]");
            }
            dos.close();
        }
        
        /**
         * <b>function:</b>读取写入点的位置信息
         * @author hoojo
         * @createDate 2011-9-23 下午05:30:29
         * @throws IOException
         */
        private void readPosInfo() throws IOException {
            DataInputStream dis = new DataInputStream(new FileInputStream(tempFile));
            int startPosLength = dis.readInt();
            startPos = new long[startPosLength];
            endPos = new long[startPosLength];
            for (int i = 0; i < startPosLength; i++) {
                startPos[i] = dis.readLong();
                endPos[i] = dis.readLong();
            }
            dis.close();
        }
        
        /**
         * <b>function:</b> 获取下载文件的长度
         * @author hoojo
         * @createDate 2011-9-26 下午12:15:08
         * @return
         */
        private int getFileSize() {
            int fileLength = -1;
            try {
                URL url = new URL(this.downloadInfo.getUrl());
                HttpURLConnection conn = (HttpURLConnection) url.openConnection();
                
                DownloadFile.setHeader(conn);
     
                int stateCode = conn.getResponseCode();
                //判断http status是否为HTTP/1.1 206 Partial Content或者200 OK
                if (stateCode != HttpURLConnection.HTTP_OK && stateCode != HttpURLConnection.HTTP_PARTIAL) {
                    LogUtils.log("Error Code: " + stateCode);
                    return -2;
                } else if (stateCode >= 400) {
                    LogUtils.log("Error Code: " + stateCode);
                    return -2;
                } else {
                    //获取长度
                    fileLength = conn.getContentLength();
                    LogUtils.log("FileLength: " + fileLength);
                }
                
                //读取文件长度
                /*for (int i = 1; ; i++) {
                    String header = conn.getHeaderFieldKey(i);
                    if (header != null) {
                        if ("Content-Length".equals(header)) {
                            fileLength = Integer.parseInt(conn.getHeaderField(i));
                            break;
                        }
                    } else {
                        break;
                    }
                }
                */
                
                DownloadFile.printHeader(conn);
            } catch (MalformedURLException e) {
                e.printStackTrace();
            } catch (IOException e) {
                e.printStackTrace();
            }
            return fileLength;
        }
    }

    这个类主要是完成读取指定url资源的内容,获取该资源的长度。然后将该资源分成指定的块数,将每块的起始下载位置、结束下载位置,分别保存在一个数组中。每块都单独开辟一个独立线程开始下载。在开始下载之前,需要创建一个临时文件,写入当前下载线程的开始下载指针位置和结束下载指针位置。

     

    五、工具类、测试类

    日志工具类

    package com.hoo.util;
     
    /**
     * <b>function:</b> 日志工具类
     * @author hoojo
     * @createDate 2011-9-21 下午05:21:27
     * @file LogUtils.java
     * @package com.hoo.util
     * @project MultiThreadDownLoad
     * @blog http://blog.csdn.net/IBM_hoojo
     * @email hoojo_@126.com
     * @version 1.0
     */
    public abstract class LogUtils {
        
        public static void log(Object message) {
            System.err.println(message);
        }
        
        public static void log(String message) {
            System.err.println(message);
        }
        
        public static void log(int message) {
            System.err.println(message);
        }
        
        public static void info(Object message) {
            System.out.println(message);
        }
        
        public static void info(String message) {
            System.out.println(message);
        }
        
        public static void info(int message) {
            System.out.println(message);
        }
    }

    下载工具类

    package com.hoo.util;
     
    import com.hoo.download.BatchDownloadFile;
    import com.hoo.entity.DownloadInfo;
     
    /**
     * <b>function:</b> 分块多线程下载工具类
     * @author hoojo
     * @createDate 2011-9-28 下午05:22:18
     * @file DownloadUtils.java
     * @package com.hoo.util
     * @project MultiThreadDownLoad
     * @blog http://blog.csdn.net/IBM_hoojo
     * @email hoojo_@126.com
     * @version 1.0
     */
    public abstract class DownloadUtils {
     
        public static void download(String url) {
            DownloadInfo bean = new DownloadInfo(url);
            LogUtils.info(bean);
            BatchDownloadFile down = new BatchDownloadFile(bean);
            new Thread(down).start();
        }
        
        public static void download(String url, int threadNum) {
            DownloadInfo bean = new DownloadInfo(url, threadNum);
            LogUtils.info(bean);
            BatchDownloadFile down = new BatchDownloadFile(bean);
            new Thread(down).start();
        }
        
        public static void download(String url, String fileName, String filePath, int threadNum) {
            DownloadInfo bean = new DownloadInfo(url, fileName, filePath, threadNum);
            LogUtils.info(bean);
            BatchDownloadFile down = new BatchDownloadFile(bean);
            new Thread(down).start();
        }
    }

    下载测试类

    package com.hoo.test;
     
    import com.hoo.util.DownloadUtils;
     
    /**
     * <b>function:</b> 下载测试
     * @author hoojo
     * @createDate 2011-9-23 下午05:49:46
     * @file TestDownloadMain.java
     * @package com.hoo.download
     * @project MultiThreadDownLoad
     * @blog http://blog.csdn.net/IBM_hoojo
     * @email hoojo_@126.com
     * @version 1.0
     */
    public class TestDownloadMain {
     
        public static void main(String[] args) {
            /*DownloadInfo bean = new DownloadInfo("http://i7.meishichina.com/Health/UploadFiles/201109/2011092116224363.jpg");
            System.out.println(bean);
            BatchDownloadFile down = new BatchDownloadFile(bean);
            new Thread(down).start();*/
            
            //DownloadUtils.download("http://i7.meishichina.com/Health/UploadFiles/201109/2011092116224363.jpg");
            DownloadUtils.download("http://mp3.baidu.com/j?j=2&url=http%3A%2F%2Fzhangmenshiting2.baidu.com%2Fdata%2Fmusic%2F1669425%2F%25E9%2599%25B7%25E5%2585%25A5%25E7%2588%25B1%25E9%2587%258C%25E9%259D%25A2.mp3%3Fxcode%3D2ff36fb70737c816553396c56deab3f1", "aa.mp3", "c:/temp", 5);
        }
    }

    多线程下载主要在第三部和第四部,其他的地方还是很好理解。源码中提供相应的注释了,便于理解。

  • 作者:hoojo
    出处:
    blog:http://blog.csdn.net/IBM_hoojo
    本文版权归作者和博客园共有,欢迎转载,但未经作者同意必须保留此段声明,且在文章页面明显位置给出原文连接,否则保留追究法律责任的权利。

版权所有,转载请注明出处 本文出自:
分享道版权所有,欢迎转载,转载请注明出处,谢谢
收藏
关注
评论
查看全文
  • 相关阅读:
    JDBC
    Oracle基本数据类型
    vue生命周期详解
    Vue的Ajax(vue-resource/axios)
    Vue实例属性/方法/生命周期
    Vue自定义指令
    深入了解组件- -- 动态组件 & 异步组件
    深入了解组件- -- 插槽
    深入了解组件- -- 自定义事件
    深入了解组件- -- Prop
  • 原文地址:https://www.cnblogs.com/hoojo/p/2196767.html
  • Copyright © 2011-2022 走看看