java相关:基于断点续传下载原理的实现

发布于 2020-7-24|标签java
复制链接
下面小妖就为大家带来一篇基于断点续传下载原理的实现。小妖觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小妖过来看看吧

需求背景

动态创建的文件下载的时候希望浏览器显示下载进度

动态创建的文件希望能够分段下载

HTTP断点续传报文

要实现HTTP断点续传必须要简单了解以下几个报文。

Accept-Ranges 告诉客户端(浏览器..)服务器端支持断点续传 服务器端返回

Range 客户端告诉服务器端从指定的的位置/范围(这里值字节数)下载资源 客户端发出

Content-Range 服务器端告诉客户端响应的数据信息,在整个返回体中本部分的字节位置 服务器端返回

ETag 资源标识 非必须 服务器端返回

Last-Modified 资源最后一次更新的时间 非必须 服务器端返回

Range 的范围格式

表示0-499个字节范围:Range: bytes=0-499

表示最后500个字节范围:Range: bytes=-500

表示500字节开始到结束范围:Range: bytes=500-

表示第一个和最后一个字节:Range: bytes=0-0,-1

表示同时指定几个范围:Range: bytes=500-600,601-999

Content-Range 的数据格式

Content-Range: bytes 0-499/22036 :表示返回0-499字节范围数据 资源一共22036个字节

原理

客户端发起请求 设置Range指定开始字节数或结束字节数 如果是从0开始也可以不用设置。

服务器端检查到客户端Range头 解析开始字节数以及结束字节数 并返回报文头 Accept-Ranges表示支持断点续传,Content-Range记录该次向客户端写入流的位置信息,然后再写入流到客户端。

服务端可以使用ETag Last-Modified 标记一下资源是否被修改。作一些验证工作,如果验证不通过则返回错误,非必须项。

java实现

OutputStream os=null; InputStream inputStream =null; File zipFile=null; try{  long zipStart=System.currentTimeMillis();  zipFile=createFile();//动态根据业务创建文件  if(logger.isInfoEnabled()){   logger.info(String.format("压缩ZIP 花费时间 %s(s) ",  (System.currentTimeMillis()-zipStart)/1000));  }  if (zipFile.exists()) {   long downloadStart=System.currentTimeMillis();   inputStream= new BufferedInputStream(new FileInputStream(zipFile));   response.reset();   os=new BufferedOutputStream(response.getOutputStream());   String userAgent = request.getHeader("USER-AGENT");   String fileName=zipFile.getName();   if (null != userAgent && -1 != userAgent.indexOf("MSIE")) {    fileName = URLEncoder.encode(fileName, "UTF8");   } else if (null != userAgent && -1 != userAgent.indexOf("Mozilla")) {    fileName = new String(fileName.getBytes("utf-8"), "ISO-8859-1");   }   response.setHeader("Accept-Ranges", "bytes");   response.setHeader("Content-Disposition",   "attachment;filename="+ fileName);   response.setContentType(MediaType.APPLICATION_OCTET_STREAM_VALUE);   long pos = 0, fileSize=zipFile.length(), last=fileSize-1;   response.setHeader("ETag",zipFile.getName().   concat(Objects.toString(fileSize))     .concat("_").concat(Objects.toString(zipFile.lastModified())));   response.setDateHeader("Last-Modified",zipFile.lastModified());   response.setDateHeader("Expires",   System.currentTimeMillis()+1000*60*60*24);   if (null != request.getHeader("Range")) {    response.setStatus(HttpServletResponse.SC_PARTIAL_CONTENT);    try {     // 暂时只处理这2种range格式 1、RANGE: bytes=111- 2、Range: bytes=0-499     String numRang = request.getHeader("Range")   .replaceAll("bytes=", "");     String[] strRange = numRang.split("-");     if (strRange.length == 2) {      pos = Long.parseLong(strRange[0].trim());      last = Long.parseLong(strRange[1].trim());     } else {      pos = Long.parseLong(numRang.replaceAll("-", "").trim());     }    } catch (NumberFormatException e) {     logger.error(request.getHeader("Range") + " error");     pos = 0;    }   }   long rangLength = last - pos + 1;   String contentRange = new StringBuffer("bytes ").   append(String.valueOf(pos)).   append("-").append(last).append("/").   append(String.valueOf(fileSize)).toString();   response.setHeader("Content-Range", contentRange);   response.addHeader("Content-Length",Objects.toString(rangLength));   if(pos>0){    inputStream.skip(pos);   }   byte[] buffer = new byte[1024*512];//每次以512KB 0.5MB的流量下载   int length = 0,sendTotal=0;   while (sendTotal < rangLength && length!=-1) {    length = inputStream.read(buffer, 0,  ((rangLength - sendTotal) <= buffer.length ?      ((int) (rangLength - sendTotal)) : buffer.length));    sendTotal = sendTotal + length;    os.write(buffer, 0, length);   }   if(os!=null){    os.flush();   }   if(logger.isInfoEnabled()){    logger.info(String.format("下载 花费时间 %s(s) ",  (System.currentTimeMillis()-downloadStart)/1000));   }  } }catch (Exception e){  if(StringUtils.endsWithIgnoreCase(e.getMessage(),"Broken pipe")){   logger.error("用户取消下载");  }  logger.error(e.getMessage(),e); }finally {  if(os!=null){   try{    os.close();   }catch (Exception e){}  }  if(inputStream!=null){   try{    IOUtils.closeQuietly(inputStream);   }catch (Exception e){}  } }}

比如google浏览器下载的时候就能看到下载进度以及暂停下载和恢复下载操作,也可以设置Range测试分段下载。

原文地址: http://www.cnblogs.com/peachyy/archive/2017/09/05/7482222.html
(0)

相关推荐

  • 【Net】StreamWriter.Write 的一点注意事项

    背景 今天在维护一个旧项目的时候,看到一个方法把string 转换为 byte[] 用的是写入内存流的,然后ToArray(),因为平常都是用System.Text.Encoding.UTF8.Get ...

  • Electron 的断点续下载

    最近用 Electron 做了个壁纸程序,需要断点续下载,在这里记录一下. HTTP断点下载相关的报文 Accept-Ranges 告诉客户端服务器是否支持断点续传,服务器返回 Content-Ran ...

  • Java对网络图片/本地图片转换成Base64编码和解码

    一.将本地图片转换成Base64编码字符串 /** * 将本地图片转换成Base64编码字符串 * * @param imgFile 图片目录路径 * @return */ public static ...

  • 钓鱼岛的土壤:基于土壤发生学原理的推测

    前言 4月26日,自然资源部在其官网上公布<钓鱼岛及其附属岛屿地形地貌调查报告>,进一步补充完善了钓鱼岛及其附属岛屿的基础地理数据体系,充分宣示了我国对钓鱼岛的主权,而且对其资源管理与生态 ...

  • Java并发多线程编程——Volatile原理与使用

    优质文章,第一时间送达 76套java从入门到精通实战课程分享 一.volitile的理解 Volatile称之为轻量级锁,被volatile修饰的变量,在线程之间是可见的. 可见即一个线程修改了这个 ...

  • 基因表达腺相关病毒载体构建技术原理

    基因表达腺相关病毒载体由腺相关病毒基因组序列和细菌质粒序列构成.细菌质粒序列含有一个Ampicillin抗性基因和一个高拷贝复制子pUCori.腺相关病毒基因组序列是从元件5'ITR开始,到元件3'I ...

  • Java中synchronized的实现原理与应用

    转自:https://blog.csdn.net/u012465296/article/details/53022317 Java中的每一个对象都可以作为锁,而在Synchronized实现同步的几种 ...

  • 聊一聊JAVA指针压缩的实现原理(图文并茂,让你秒懂)

    转自:https://blog.csdn.net/liujianyangbj/article/details/108049482 前言 在网上大家很多都看到过这样一句话:"JVM内存最好不要 ...

  • 【Java面试题第八期】不会还有人不懂Java最基本的异常处理原理吧?

    Java使用面向对象的方式来处理异常,它把程序中发生的每个异常也都分别封装到一个对象来表示的,该对象中包含有异常的信息. Java对异常进行了分类,不同类型的异常分别用不同的 Java 类表示,所有异 ...

  • 15个经典面试问题,java中异常处理机制的原理和应用

    微服务架构 第1章 微服务概述 什么是微服务 常见的微服务组件 常用的微服务框架 微服务架构设计模式 如何实施微服务 从微服务的起源和现实业务的角度探讨微服务 第2章 微服务设计原则 设计原则之分层架 ...

  • DMD相关动物模型构建技术原理

    DMD这种疾病最广泛使用的动物模型是30年前首次发现的肌营养不良蛋白缺陷mdx小鼠,它像DMD患者一样,在抗肌萎缩蛋白基因本身上存在突变,从而导致全功能性的肌营养不良蛋白表达减少.小鼠体内的Dmd m ...

  • 基于工作记忆的原理谈如何记笔记

    发布时间:2021-05-22 文:陈佳昕 来源:泡泡心理(ID:psychologiacal_bread) 无论你是学生还是在职人士,在学习新东西的时候可能都用过记笔记的方式.你在记笔记的时候是否有 ...