`
lishumingwm163.com
  • 浏览: 334526 次
  • 性别: Icon_minigender_1
  • 来自: 济南
社区版块
存档分类
最新评论

http断点续传

    博客分类:
  • java
阅读更多

Http文件下载的普通模式就不多说了,断点下载与普通模式不一样的是,断点下载的头信息里面增加了一个属性
RANGE: bytes=100000-
这里RANGE代表客户端要从那个位置开始下载

 

而服务器返回时和普通模式不同的是:
1:多了一个属性
Content-Range=bytes 100000-19999/20000
注意还有一些属性要设置,和普通模式一样,例如Content-Length属性
2:返回码为206

 

然后我们来看一段实际应用中的断点下载代码,注意这里我使用的是SpringMVC模式开发的:

Java代码  收藏代码
  1. @SuppressWarnings({ "unchecked" })  
  2. @RequestMapping(value = "/downOdex.do")  
  3. public ResponseEntity<String> downFile(  
  4.         @RequestParam(value="odexName")String odexName,  
  5.         HttpServletResponse response,  
  6.         HttpServletRequest request){  
  7.     InputStream inputStream = null;  
  8.     ServletOutputStream out = null;  
  9.     try {  
  10.         File file = new File(OdexManage.odexFileBasePath + "\\" + odexName);  
  11.         int fSize = Integer.parseInt(String.valueOf(file.length()));    
  12.         response.setCharacterEncoding("utf-8");  
  13.         response.setContentType("application/x-download");    
  14.         response.setHeader("Accept-Ranges""bytes");    
  15.         response.setHeader("Content-Length", String.valueOf(fSize));    
  16.         response.setHeader("Content-Disposition""attachment;fileName=" + odexName);  
  17.         inputStream=new FileInputStream(OdexManage.odexFileBasePath + "\\" + odexName);  
  18.         long pos = 0;    
  19.         if (null != request.getHeader("Range")) {  
  20.             // 断点续传  
  21.             response.setStatus(HttpServletResponse.SC_PARTIAL_CONTENT);    
  22.             try {    
  23.                 pos = Long.parseLong(request.getHeader("Range").replaceAll("bytes=""").replaceAll("-"""));    
  24.             } catch (NumberFormatException e) {  
  25.                 pos = 0;    
  26.             }    
  27.         }    
  28.         out = response.getOutputStream();    
  29.         String contentRange = new StringBuffer("bytes ").append(pos+"").append("-").append((fSize - 1)+"").append("/").append(fSize+"").toString();  
  30.         response.setHeader("Content-Range", contentRange);    
  31.         inputStream.skip(pos);    
  32.         byte[] buffer = new byte[1024*10];  
  33.         int length = 0;    
  34.         while ((length = inputStream.read(buffer, 0, buffer.length)) != -1) {    
  35.             out.write(buffer, 0, length);  
  36.         <span style="font-family: Consolas;">Thread.sleep(</span><code class="java value">100</code><code class="java plain">);</code>  
  37.         }  
  38.     } catch (Exception e) {  
  39.         logger.error("ODEX软件下载异常:"+e);  
  40.     }finally{  
  41.          try {  
  42.              if(null != out) out.flush();  
  43.              if(null != out) out.close();  
  44.              if(null != inputStream) inputStream.close();   
  45.         } catch (IOException e) {  
  46.         }  
  47.     }  
  48.     return new ResponseEntity(null,HttpStatus.OK);  
  49. }  

 其重点在于HTTP协议里面属性有一些不同的地方,还有就是InputStream跳过不需要读的文件,和注意关闭流。
通过核心代码也可以看到,其实这和是不是SpringMVC没多大关系,所以你可以很容易的应用到你的项目中。

 

例如对于这段代码,我访问如下路径
http://localhost/api/downOdex.do?odexName=D03BFBAE35BEC791092E52EC907D1F69.ZIP

使用浏览器自带的下载工具进行下载,这样可以控制下载和暂停,来观察断点下载的过程。

点击暂停会看到下载暂停下来

可以选择继续下载。

过程中你会看到,客户端不会马上把文件下载下来,而是慢慢下载,就是因为我在程序中增加了停顿来查看这个下载过程

Java代码  收藏代码
  1. Thread.sleep(100);  

 然后我们的缓冲区是

Java代码  收藏代码
  1. byte[] buffer = new byte[1024*10];  

 所以,每秒的速度就是100KB理论速度,因为是本地,所以接近100KB每秒

分享到:
评论

相关推荐

Global site tag (gtag.js) - Google Analytics