关于下载文件,首先需要了解一点io和http请求,线程等相关知识,然后才能一步一步的趴坑整理出一个较为满意的结果。原先我写过一个较为简单的下载是单线程的,如下

package cn.zectec.hamster.baseservice.utils;

import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.util.StringUtils;

import javax.servlet.ServletOutputStream;
import javax.servlet.http.HttpServletResponse;
import java.io.*;
import java.net.*;
import java.util.HashMap;
import java.util.Map;

public class DownloadUtil {
   private static Logger logger = LoggerFactory.getLogger(DownloadUtil.class);

   public static void errorFilePath(HttpServletResponse response, String msg){
      //没有该消息记录直接返回无此文件
      try {
         ServletOutputStream out=response.getOutputStream();
         OutputStreamWriter ow=new OutputStreamWriter(out,"UTF-8");
         ow.write(msg);
         ow.flush();
         ow.close();
      } catch (IOException e) {

      }catch (Exception e) {
         e.printStackTrace();
      }

   }
   /**
    * @param filePath 要下载的文件路径
    * @param returnName 返回的文件名
    * @param response HttpServletResponse
    * @param delFlag 是否删除文件
    */
   protected void download(String filePath,String returnName,HttpServletResponse response,boolean delFlag){
      prototypeDownload(new File(filePath), returnName, response, delFlag);
   }


   /**
    * @param file 要下载的文件
    * @param returnName 返回的文件名
    * @param response HttpServletResponse
    * @param delFlag 是否删除文件
    */
   public static void download(File file, String returnName, HttpServletResponse response, boolean delFlag){
      prototypeDownload(file, returnName, response, delFlag);
   }
   
   /**
    * @param file 要下载的文件
    * @param returnName 返回的文件名
    * @param response HttpServletResponse
    * @param delFlag 是否删除文件
    */
   private static void prototypeDownload(File file, String returnName, HttpServletResponse response, boolean delFlag){
      // 下载文件
      FileInputStream inputStream = null;
      ServletOutputStream outputStream = null;
      try {
         if(!file.exists()) return;
         response.reset();
         //设置响应类型   PDF文件为"application/pdf",WORD文件为:"application/msword", EXCEL文件为:"application/vnd.ms-excel"。
         response.setContentType("application/octet-stream;charset=utf-8");
         //设置响应的文件名称,并转换成中文编码
         returnName = URLEncoder.encode(returnName,"UTF-8");
         returnName = response.encodeURL(new String(returnName.getBytes(),"iso8859-1"));    //保存的文件名,必须和页面编码一致,否则乱码

         //attachment作为附件下载;inline客户端机器有安装匹配程序,则直接打开;注意改变配置,清除缓存,否则可能不能看到效果
         response.addHeader("Content-Disposition",   "attachment;filename="+returnName);

         //将文件读入响应流
         inputStream = new FileInputStream(file);
         outputStream = response.getOutputStream();
         int length = 1024;
         int readLength=0;
         byte buf[] = new byte[1024];
         readLength = inputStream.read(buf, 0, length);
         while (readLength != -1) {
            outputStream.write(buf, 0, readLength);
            readLength = inputStream.read(buf, 0, length);
         }
      }catch (IOException e) {
         errorFilePath(response,"无此文件");
      }catch (Exception e) {
         e.printStackTrace();
      }finally {
         try {
            outputStream.flush();
         } catch (IOException e) {

         }
         try {
            outputStream.close();
         } catch (IOException e) {

         }
         try {
            inputStream.close();
         } catch (IOException e) {

         }
         //删除原文件

         if(delFlag) {
            file.delete();
         }
      }
   }

   /**
    * by tony 2013-10-17
    * @param byteArrayOutputStream 将文件内容写入ByteArrayOutputStream
    * @param response HttpServletResponse 写入response
    * @param returnName 返回的文件名
    */
   public void download(ByteArrayOutputStream byteArrayOutputStream, HttpServletResponse response, String returnName) throws IOException{
      response.setContentType("application/octet-stream;charset=utf-8");
      returnName = response.encodeURL(new String(returnName.getBytes(),"iso8859-1"));          //保存的文件名,必须和页面编码一致,否则乱码
      response.addHeader("Content-Disposition",   "attachment;filename=" + returnName);  
      response.setContentLength(byteArrayOutputStream.size());
      
      ServletOutputStream outputstream = response.getOutputStream(); //取得输出流
      byteArrayOutputStream.writeTo(outputstream);               //写到输出流
      byteArrayOutputStream.close();                         //关闭
      outputstream.flush();                                //刷数据
   }

   /***
    * 浏览器远程下载文件
    */
   public static void downloadFileByHttpUrl(HttpServletResponse response,String urlStr,String fileName){
      OutputStream out = null;
      InputStream ips = null;
      try {
         logger.info("第一次请求url==>" + urlStr);
         URL url = new URL(urlStr);
         HttpURLConnection httpUrlConn = (HttpURLConnection) url.openConnection();
         httpUrlConn.addRequestProperty("Content-type", "application/json");
         //设置请求超时时间
         httpUrlConn.setReadTimeout(15000);
         httpUrlConn.setConnectTimeout(15000);
         httpUrlConn.setDoOutput(true); //允许写出
         httpUrlConn.setDoInput(true);//允许写入
         httpUrlConn.setUseCaches(false);//不使用缓存
         // 设置请求方式(GET/POST)
         httpUrlConn.setRequestMethod("GET");
         httpUrlConn.connect();
         int responseCode = httpUrlConn.getResponseCode();
         if(responseCode!=200){
            errorFilePath(response,"无此文件");
         }else {
            ips = httpUrlConn.getInputStream();//字节流 输入流
            response.reset();
            response.setContentType("application/x-download");
            response.addHeader("Content-Disposition","attachment;filename="+ new String(fileName.getBytes(),"iso-8859-1"));
            response.setContentType("application/octet-stream");
            out = new BufferedOutputStream(response.getOutputStream());
            int length = 1024;
            int readLength=0;
            byte buf[] = new byte[1024];
            readLength = ips.read(buf, 0, length);
            while (readLength != -1) {
               out.write(buf, 0, readLength);
               readLength = ips.read(buf, 0, length);
            }
         }
      }catch (IOException e) {
         errorFilePath(response,"无此文件");
      }catch (Exception e) {
         e.printStackTrace();
      }finally {
         try {
            out.flush();
         } catch (IOException e) {

         }
         try {
            out.close();
         } catch (IOException e) {

         }
         try {
            ips.close();
         } catch (IOException e) {

         }
      }
   }

   /***
    * http 请求获取下载文件保存到本地
    * urlStr 远程地址
    * filePath 指定本地保存路径
    */
   public static Map<String,Object> downloadFileByHttpUrl(String urlStr, String filePath){
      Map<String,Object> map = new HashMap<>();
      map.put("success",false);
      OutputStream out = null;
      InputStream ips = null;
      String requestMethod="GET";
      try {
         File file = new File(filePath);
         logger.info("===================:"+urlStr);
         logger.info("===================获取流媒体服务中的文件:"+filePath);
         URL url = new URL(urlStr);
         HttpURLConnection httpUrlConn = (HttpURLConnection) url.openConnection();
         httpUrlConn.addRequestProperty("Content-type", "application/json");
         //httpUrlConn.addRequestProperty("accessToken",accessToken);
         //设置请求超时时间
         httpUrlConn.setReadTimeout(30000);
         httpUrlConn.setConnectTimeout(30000);
         httpUrlConn.setDoOutput(true); //允许写出
         httpUrlConn.setDoInput(true);//允许写入
         httpUrlConn.setUseCaches(false);//不使用缓存

         // 设置请求方式(GET/POST)
         httpUrlConn.setRequestMethod(requestMethod);
         if ("GET".equalsIgnoreCase(requestMethod)) {
            httpUrlConn.connect();
         }
         // 将返回的输入流转换成字符串
         ips = httpUrlConn.getInputStream();
         out = new FileOutputStream(file,false);
         int length = 1024;
         int readLength=0;
         byte buf[] = new byte[1024];
         readLength = ips.read(buf, 0, length);
         while (readLength != -1) {
            String s = new String(buf, "UTF-8");
            if (s.contains("未找到")) {
               logger.info("==================putIOToResponseByHttp错误信息:"+s);
               file.delete();
               map.put("msg",s);
               return map;
            }else{
               out.write(buf, 0, readLength);
               readLength = ips.read(buf, 0, length);
            }
         }
         map.put("success",true);
      }catch (IOException e) {
         logger.info("无此文件");
      }catch (Exception e) {
         e.printStackTrace();
      }finally {
         try {
            out.flush();
         } catch (IOException e) {

         }
         try {
            out.close();
         } catch (IOException e) {

         }
         try {
            ips.close();
         } catch (IOException e) {

         }
      }
      return map;
   }

   /***
    * 根据远程http地址将流存到respons 返回失败或者成功
    * @param response 浏览器response
    * @param urlStr 远程http地址
    * @return 返回失败或者成功
    */
   public static boolean putIOToResponseByHttp(HttpServletResponse response, String urlStr) {
      OutputStream out = null;
      InputStream ips = null;
      String requestMethod="GET";
      try {
         if (StringUtils.isEmpty(urlStr)) {
            return false;
         }else{
               URL url = new URL(urlStr);
               HttpURLConnection httpUrlConn = (HttpURLConnection) url.openConnection();
               httpUrlConn.addRequestProperty("Content-type", "application/json");
               //httpUrlConn.addRequestProperty("accessToken",accessToken);
               //设置请求超时时间
               httpUrlConn.setReadTimeout(300000);
               httpUrlConn.setConnectTimeout(300000);
               httpUrlConn.setDoOutput(true); //允许写出
               httpUrlConn.setDoInput(true);//允许写入
               httpUrlConn.setUseCaches(false);//不使用缓存

               // 设置请求方式(GET/POST)
               httpUrlConn.setRequestMethod(requestMethod);
               if ("GET".equalsIgnoreCase(requestMethod)) {
                  httpUrlConn.connect();
               }
               // 将返回的输入流转换成字符串
               ips = httpUrlConn.getInputStream();
               out = new BufferedOutputStream(response.getOutputStream());
               int length = 1024;
               int readLength=0;
               byte buf[] = new byte[1024];
               readLength = ips.read(buf, 0, length);
               boolean bool = true;
               while (readLength != -1) {
                  if (bool && new String(buf, "UTF-8").contains("未找到")) {
                     logger.info("==================putIOToResponseByHttp错误信息:"+new String(buf, "UTF-8"));
                     return false;
                  }else{
                     bool = false;
                     out.write(buf, 0, readLength);
                     readLength = ips.read(buf, 0, length);
                  }
               }
            }
            return true;
      }catch (IOException e) {
         errorFilePath(response,"无此文件");
      }catch (Exception e) {
         e.printStackTrace();
      }finally {
         try {
            out.flush();
         } catch (IOException e) {

         }
         try {
            out.close();
         } catch (IOException e) {

         }
         try {
            ips.close();
         } catch (IOException e) {

         }
      }
      return false;
   }
   public static InputStream returnIoByHttUrl(String requestMethod,String urlStr){
      try {
         URL url = new URL(urlStr);
         HttpURLConnection httpUrlConn = (HttpURLConnection) url.openConnection();
         httpUrlConn.addRequestProperty("Content-type", "application/json");
         //设置请求超时时间
         httpUrlConn.setReadTimeout(30000);
         httpUrlConn.setConnectTimeout(30000);
         httpUrlConn.setDoOutput(true); //允许写出
         httpUrlConn.setDoInput(true);//允许写入
         httpUrlConn.setUseCaches(false);//不使用缓存

         // 设置请求方式(GET/POST)
         httpUrlConn.setRequestMethod(requestMethod);
         if ("GET".equalsIgnoreCase(requestMethod)) {
            httpUrlConn.connect();
         }
         // 将返回的输入流转换成字符串
         return httpUrlConn.getInputStream();
      }catch (IOException e) {
         logger.info("无此文件");
      }catch (Exception e) {
         e.printStackTrace();
      }
      return null;
   }


   /***
    * 校验远程地址是否可用
    * @param requestUrl 请求地址
    * @return 是否可用
    */
   public static Boolean checkIP(String requestUrl){
      try {
         String hostAddress = InetAddress.getLocalHost().getHostAddress();
         logger.info("java获取到的当前页面地址"+hostAddress);
         URL url = new URL(requestUrl);
         HttpURLConnection httpUrlConn = (HttpURLConnection) url.openConnection();
         httpUrlConn.addRequestProperty("Content-type", "application/json");
         //设置请求超时时间
         httpUrlConn.setReadTimeout(5000);
         httpUrlConn.setConnectTimeout(5000);
         httpUrlConn.setDoOutput(true); //允许写出
         httpUrlConn.setDoInput(true);//允许写入
         httpUrlConn.setUseCaches(false);//不使用缓存
         // 设置请求方式(GET/POST)
         httpUrlConn.setRequestMethod("GET");
         httpUrlConn.connect();
         int responseCode = httpUrlConn.getResponseCode();
         if(responseCode==200){
            logger.info("=============="+requestUrl+":地址有效");
            return true;
         }
      }catch (Exception e){
         return false;
      }
      return false;
   }


   /***
    * 校验远程文件是否可用
    * @return 是否可用
    */
   public static Boolean checkFile(String urlStr){
      InputStream ips = null;
      String requestMethod="GET";
      try {
         URL url = new URL(urlStr);
         HttpURLConnection httpUrlConn = (HttpURLConnection) url.openConnection();
         httpUrlConn.addRequestProperty("Content-type", "application/json");
         //httpUrlConn.addRequestProperty("accessToken",accessToken);
         //设置请求超时时间
         httpUrlConn.setReadTimeout(30000);
         httpUrlConn.setConnectTimeout(30000);
         httpUrlConn.setDoOutput(true); //允许写出
         httpUrlConn.setDoInput(true);//允许写入
         httpUrlConn.setUseCaches(false);//不使用缓存

         // 设置请求方式(GET/POST)
         httpUrlConn.setRequestMethod(requestMethod);
         if ("GET".equalsIgnoreCase(requestMethod)) {
            httpUrlConn.connect();
         }
         // 将返回的输入流转换成字符串
         ips = httpUrlConn.getInputStream();
         int length = 1024;
         int readLength=0;
         byte buf[] = new byte[1024];
         readLength = ips.read(buf, 0, length);
         while (readLength != -1) {
            String s = new String(buf, "UTF-8");
            if (s.contains("未找到")) {
               return false;
            }else{
               return true;
            }
         }
      }catch (IOException e) {
         logger.info("无此文件");
      }catch (Exception e) {
         e.printStackTrace();
      }finally {
         try {
            ips.close();
         } catch (IOException e) {

         }
      }
      return false;
   }



}

用于单线程下载遇到大文件时读写过于缓慢,查了半天的百度收罗了很多知识,最后整理成一个较为完整的多线程下载工具包

总共三个文件:

One:DownloadFileWithThreadPool

package cn.zectec.hamster.videorecord.download;

import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import java.io.File;
import java.io.IOException;
import java.net.HttpURLConnection;
import java.net.URL;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;

public class DownloadFileWithThreadPool {

    private static Logger logger = LoggerFactory.getLogger(DownloadFileWithThreadPool.class);

    public void getFileWithThreadPoolByHttp(String urlLocation, String filePath, int poolLength) throws IOException {
        //有顺序的线程
        ExecutorService threadPool = Executors.newSingleThreadExecutor();

        long len = getContentLength(urlLocation);
        for (int i = 0; i < poolLength; i++) {
            long start = i * len / poolLength;
            long end = (i + 1) * len / poolLength - 1;
            if (i == poolLength - 1) {
                end = len;
            }
            DownloadWithRangeByHttpUrl download = new DownloadWithRangeByHttpUrl(urlLocation, filePath, start, end);
            threadPool.execute(download);
        }
        threadPool.shutdown();
    }

    public void getFileWithThreadPoolByLocal(String localAddress, String filePath, int poolLength) throws IOException {
        //有顺序的线程
        ExecutorService threadPool = Executors.newSingleThreadExecutor();
        File file = new File(localAddress);
        if(!file.exists() || !file.isFile()){
            logger.info("多线程下载本地文件:文件不存在");
            return;
        }
        long len = new File(localAddress).length();
        for (int i = 0; i < poolLength; i++) {
            long start = i * len / poolLength;
            DownloadWithRangeByLocalAddress download = new DownloadWithRangeByLocalAddress(localAddress, filePath, len,i,start);
            threadPool.execute(download);
        }
        threadPool.shutdown();
    }

    public static long getContentLength(String urlLocation) throws IOException {
        URL url = null;
        if (urlLocation != null) {
            url = new URL(urlLocation);
        }
        HttpURLConnection conn = (HttpURLConnection) url.openConnection();
        conn.setReadTimeout(5000);
        conn.setRequestMethod("GET");
        long len = conn.getContentLength();

        return len;
    }

}

Two:DownloadWithRangeByHttpUrl

package cn.zectec.hamster.videorecord.download;

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

/***
 * 请求远程http地址,指定文件地址下载文件
 */
public class DownloadWithRangeByHttpUrl implements Runnable {

    /**远程调用地址*/
    private String urlLocation;

    /**文件保存地址*/
    private String filePath;

    /**下载起始位置*/
    private long start;

    /**下载结束位置*/
    private long end;

    DownloadWithRangeByHttpUrl(String urlLocation, String filePath, long start, long end) {
        this.urlLocation = urlLocation;
        this.filePath = filePath;
        this.start = start;
        this.end = end;
    }

    @Override
    public void run() {
        try {
            HttpURLConnection conn = getHttp();
            conn.setRequestProperty("Range", "bytes=" + start + "-" + end);

            File file = new File(filePath);
            RandomAccessFile out = null;
            if (file != null) {
                out = new RandomAccessFile(file, "rw");
            }
            out.seek(start);
            InputStream in = conn.getInputStream();
            byte[] b = new byte[1024];
            int len = 0;
            while ((len = in.read(b)) >= 0) {
                out.write(b, 0, len);
            }
            in.close();
            out.close();
        } catch (Exception e) {
            e.getMessage();
        }

    }

    public HttpURLConnection getHttp() throws IOException {
        URL url = null;
        if (urlLocation != null) {
            url = new URL(urlLocation);
        }
        HttpURLConnection conn = (HttpURLConnection) url.openConnection();
        //设置请求超时时间
        conn.setReadTimeout(30000);
        conn.setConnectTimeout(30000);
        conn.setDoOutput(true); //允许写出
        conn.setDoInput(true);//允许写入
        conn.setUseCaches(false);//不使用缓存
        conn.setRequestMethod("GET");

        return conn;
    }

}

Three:DownloadWithRangeByLocalAddress

package cn.zectec.hamster.videorecord.download;

import sun.misc.Cleaner;
import sun.nio.ch.DirectBuffer;

import java.io.File;
import java.io.RandomAccessFile;
import java.nio.MappedByteBuffer;
import java.nio.channels.FileChannel;

/***
 * 获取本地资源路径,指定文件地址保存文件
 */
public class DownloadWithRangeByLocalAddress implements Runnable {

    // 缓冲区大小为3M
    final int BUFFER_SIZE = 0x300000;

    /**本地文件地址*/
    private String localAddress;

    /**文件保存地址*/
    private String filePath;

    /**如index/10,文件拆分为十段,index为第几段*/
    private long index;

    /**下载起始位置*/
    private long start;

    /**文件大小*/
    private long len;

    DownloadWithRangeByLocalAddress(String localAddress, String filePath, long len,long index,long start) {
        this.localAddress = localAddress;
        this.filePath = filePath;
        this.index = index;
        this.start = start;
        this.len = len;
    }

    @Override
    public void run() {
        try {
            File sourcefile = new File(localAddress);
            RandomAccessFile randomAccessFile = new RandomAccessFile(sourcefile, "r");
            FileChannel channel = randomAccessFile.getChannel();
            MappedByteBuffer inputBuffer = channel.map(FileChannel.MapMode.READ_ONLY, len*index/10,len/10);

            File file = new File(filePath);
            RandomAccessFile out = null;
            if (file != null) {
                out = new RandomAccessFile(file, "rw");
            }
            out.seek(start);
            byte[] dst = new byte[BUFFER_SIZE];// 每次读出3M的内容

            for (int offset = 0; offset < inputBuffer.capacity(); offset += BUFFER_SIZE) {
                if (inputBuffer.capacity() - offset >= BUFFER_SIZE) {

                    for (int i = 0; i < BUFFER_SIZE; i++) {
                        dst[i] = inputBuffer.get(offset + i);
                    }

                } else {

                    for (int i = 0; i < inputBuffer.capacity() - offset; i++) {
                        dst[i] = inputBuffer.get(offset + i);
                    }

                }
                out.write(dst, 0, BUFFER_SIZE);
            }
            out.close();
            inputBuffer.force();
            channel.force(true);
            channel.close();
            randomAccessFile.close();
            unmap(inputBuffer);
        } catch (Exception e) {
            e.getMessage();
        }

    }
    private void unmap(MappedByteBuffer var0) {
        Cleaner var1 = ((DirectBuffer)var0).cleaner();
        if (var1 != null) {
            var1.clean();
        }
    }
}

解说:

 DownloadFileWithThreadPool 该类用于创建线程池并执行相应的任务

里面有两个方法 通过调用这两个方法 来实现多线程下载 远程文件 或者本地文件到指定文件夹下

getFileWithThreadPoolByHttp(String urlLocation, String filePath, int poolLength)

这个方法根据参数说明:

urlLocation 视频远程访问地址

filePath 文件保存的绝对路径

poolLength 线程数

getFileWithThreadPoolByLocal(String localAddress, String filePath, int poolLength)

这个方法根据参数说明:

localAddress 本地文件路径

filePath 文件保存的绝对路径

poolLength 线程数

 

使用:

在最后我只写一下我是怎么使用的,其实调用非常简单只要传对参数就行了

远程下载请求示例:

long time = System.currentTimeMillis();
DownloadFileWithThreadPool pool = new DownloadFileWithThreadPool();
try {
    pool.getFileWithThreadPoolByHttp(urlLocation, filePath, 10);
    System.out.println("下载成功:"+( System.currentTimeMillis() - time )+"毫秒");
} catch (IOException e) {
    e.printStackTrace();
    System.out.println("服务器出错");
}

下载本地文件请求示例

long time = System.currentTimeMillis();
DownloadFileWithThreadPool pool = new DownloadFileWithThreadPool();
try {
    pool.getFileWithThreadPoolByLocal(urlLocation, filePath, 10);
    System.out.println("下载成功:"+( System.currentTimeMillis() - time )+"毫秒");
} catch (IOException e) {
    e.printStackTrace();
    System.out.println("服务器出错");
}