springboot下载文件 范围下载

关键词:springboot,download,Range,Content-Range,Content-Length,http code 206 Partial Content

下载文件的一部分,我们在request header:Range 中指定要获取的文件的字节范围。要注意http response header:Content-Length 一定要与Range中所表示的获取bytes的范围长度相等。

如果Content-Length大于要Range所表示的bytes长度,那么客户端在发起请求后,下载完相应的Range范围的bytes后,连接不会自动关闭,直到连接超时,抛出异常。

如果Content-Length小于要Range所表示的bytes长度,那么发起请求后,只会获得Content-Length所指定的较小范围的bytes,导致获取的数据不完整。

controller:

    @SneakyThrows
@GetMapping("/download")
public void download(HttpServletResponse response,
@RequestParam String versionNum,
@RequestHeader(value = HttpHeaders.RANGE, required = false) String range) {
if (StringUtils.isEmpty(versionNum)) {
throw new ServiceException("版本号不能为空");
}
// jar包的上一级目录
String canonicalPath = FileUtil.getCanonicalPath(new File(".."));
String filePath = canonicalPath + "/device-upgrade-pack-repo/" + versionNum + "/" + versionNum + ".tar";
long fileSize = FileUtil.file(filePath).length(); // 获得文件流
BufferedInputStream inputStream = FileUtil.getInputStream(filePath); // offset表示从哪个字节开始读取,length表示读取多少个字节
long offset = 0;
long length;
// 如果range为空,那么就是读取整个文件
if (StringUtils.isEmpty(range)) {
length = fileSize;
} else {
// 根据range计算offset和length
Pair<Long, Long> pair = getOffset(range);
offset = pair.getLeft();
length = pair.getRight();
// length == -1 表示 end range 为最后一个字节
if (length == -1) {
length = fileSize - offset + 1;
}
// 跳过文件的offset个字节
inputStream.skip(offset);
} response.setContentType("application/octet-stream");
response.setHeader("Content-disposition",
"attachment;filename=" + versionNum + ".tar");
response.setHeader("Content-Length", String.valueOf(length));
response.setHeader("Content-Range", "bytes" + " " +
offset + "-" + (offset + length - 1) + "/" + fileSize);
response.setStatus(HttpServletResponse.SC_PARTIAL_CONTENT);
ServletOutputStream os = response.getOutputStream(); byte[] buf = new byte[1024];
int bytesRead;
// sum 是当前读取的字节数
int sum = 0;
while (sum < length && (bytesRead = inputStream.read(buf, 0, (length - sum) <= buf.length ? (int) (length - sum) : buf.length)) != -1) {
os.write(buf, 0, bytesRead);
sum += bytesRead;
} inputStream.close();
os.close();
}

解析range:

    /**
* left is offset, right is length
*
* @param range
* @return
*/
private Pair<Long, Long> getOffset(String range) {
if (!range.contains("bytes=") || !range.contains("-")) {
throw new ServiceException("range 格式错误");
}
range = range.substring(range.lastIndexOf("=") + 1).trim();
String[] split = range.split("-");
if (split.length == 0 || split.length > 2) {
throw new ServiceException("range 格式错误");
}
if (split.length == 1) {
if (range.startsWith("-")) {
return Pair.of(0L, Long.parseLong(split[0]) + 1);
} else {
// -1 表示length无限制
return Pair.of(Long.parseLong(split[0]), -1L);
}
} else {
return Pair.of(Long.parseLong(split[0]), Long.parseLong(split[1]) - Long.parseLong(split[0]) + 1);
}
}

测试下载接口:

    @Test
void downloadObject() {
OkHttpClient httpClient = HttpUtil.getHttpClient();
Request request = new Request.Builder()
.get()
.url("http://localhost:8081/device/version/download?versionNum=1.1.1.2023")
.header("Range", "bytes=0-10")
.build(); try (Response response = httpClient.newCall(request).execute()) {
InputStream inputStream = response.body().byteStream();
byte[] buf = new byte[5];
int readBytes; while ((readBytes = inputStream.read(buf)) != -1) {
if (readBytes < buf.length) {
buf = Arrays.copyOf(buf, readBytes);
}
System.out.println(Arrays.toString(buf) + " " + readBytes);
}
inputStream.close();
response.close();
System.out.println("end");
} catch (IOException e) {
throw new ServiceException("请求异常");
}
} 输出结果:
[49, 46, 49, 46, 49] 5
[46, 50, 48, 50, 51] 5
[46] 1
end

springboot下载文件 范围下载的更多相关文章

  1. 使用SpringBoot实现文件的下载

    上一篇博客:使用SpringBoot实现文件的上传 已经实现了文件的上传,所以紧接着就是下载 首先还是html页面的简单设计 <form class="form-signin" ...

  2. django 中下载文件与下载保存为excel

    一.django 中下载文件 在实际的项目中很多时候需要用到下载功能,如导excel.pdf或者文件下载,当然你可以使用web服务自己搭建可以用于下载的资源服务器,如nginx,这里我们主要介绍dja ...

  3. php 下载文件/直接下载数据内容

    思路步骤 * 定义参数 * 魔术方法 * 执行下载 * 获取设置属性函数 * 获取设置文件mime 类型 * 获取设置下载文件名 * 设置header * 下载函数 实现代码 class DownFi ...

  4. 01_基于TCP的循环为同一个客户端下载文件的下载器

    原版: TCP分为客户端(client)和服务器(server),每次服务器只能为客户端提供一次的下载服务. 改良版: TCP分为客户端(client)和服务器(server), (1)每次服务器能为 ...

  5. AFHTTPSessionManager下载文件 及下载中 进度条处理,进度条处理需要特别注意,要加载NSRunLoop 中

    1.下载文件 和进度条处理代码 - (void)timer:(NSTimer *)timer{ // 另一个View中 进度条progress属性赋值 _downloadView.progress = ...

  6. c#导出文件,下载文件,命名下载后的文件名

    Page.Response.AppendHeader("Content-Disposition", "attachment;filename=" + HttpU ...

  7. selenium 下载文件设置下载路径

    Chrome 文件下载 Chrome浏览器类似,设置其options: download.default_directory:设置下载路径 profile.default_content_settin ...

  8. UWP 下载文件显示下载进度

    <Page x:Class="WgscdProject.TestDownloadPage" xmlns="http://schemas.microsoft.com/ ...

  9. LoadRunner上传及下载文件

    (1)LoadRunner上传文件 web_submit_data("importStudent.do", "Action=https://testserver/cons ...

  10. python 通过ntlm验证下载文件

    最近使用python实现一个小工具,需要从网站下载文件,下载时服务端需要进行ntlm验证,否则返回401错误响应.经研究 requests库配合 requests-ntlm 可以解决这个问题. ntl ...

随机推荐

  1. PHP反序列化常用魔术方法

    PHP反序列化 php序列化(serialize):是将变量转换为可保存或传输的字符串的过程 php反序列化(unserialize):就是在适当的时候把这个字符串再转化成原来的变量使用 PHP反序列 ...

  2. Github疯传!谷歌师兄的LeetCode刷题笔记开源了!

    有小伙伴私聊我说刚开始刷LeetCode的时候,感到很吃力,刷题效率很低.我以前刷题的时候也遇到这个问题,直到后来看到这个谷歌师兄总结的刷题笔记,发现LeetCode刷题都是套路呀,掌握这些套路之后, ...

  3. element-ui Tabs 标签页刷新页面状态不丢失

    element-ui Tabs 标签页刷新页面状态不丢失 转载请表明出处 https://www.cnblogs.com/niexianda/p/14765111.html 效果 一般在使用Tabs组 ...

  4. vue2中v-if 或者 v-show 使用数组中的值判断不生效

    知识点来源:博客园==> 外号蓝大胖// 对象this.$set(obj, key, value)/vue.set(obj, key, value)// 数组this.$set(arr, ind ...

  5. 【python基础】函数-参数形式

    鉴于函数定义中可能包含多个形参变量,因此函数调用中也可能包含多个实参变量.向函数传递实参变量给形参变量的方式有很多,可使用位置参数,这要求实参变量的顺序与形参变量的顺序相同:也可使用关键字参数,都由变 ...

  6. 洛谷 P5540 [BalkanOI2011] timeismoney | 最小乘积生成树

    题意 给一个无向图,边有两个权 \(a\) 和 \(b\),定义一个生成树的权值是 \(\left(\sum\limits_{e\in T}a_e\right)\left(\sum\limits_{e ...

  7. 【python基础】类-继承

    编写类时,并非总是要从空白开始.如果要编写的类时另一个现成类的特殊版本,可使用继承.一个类继承另一个类时,它将自动获得另一个类的所有属性和方法 原有的类称为父类,而新类被称为子类.子类继承了其父类的所 ...

  8. 10分钟讲清int 和 Integer 的区别

    其实在Java编程中,int和Integer都是非常常用的数据类型,但它们之间存在一些关键的区别,特别是在面向对象编程中.所以接下来,就让我们一起来探讨下关于int和Integer的区别这个问题吧. ...

  9. 磐舟磐基平台:基于KubeEdge的落地实践

    摘要:实现统一管理.简化多集群的运维系统.减少运营成本:同时也成功将前面提到的500台鲲鹏服务器以及它上面的BC Linux for Euler集群纳入磐基PaaS平台的大家庭之中,运维效率大幅增加. ...

  10. Blazor前后端框架Known-V1.2.1

    V1.2.1 Known是基于C#和Blazor开发的前后端分离快速开发框架,开箱即用,跨平台,一处代码,多处运行. 概述 基于C#和Blazor实现的快速开发框架,前后端分离,开箱即用. 跨平台,单 ...