简介:
大家应该都听说过分片上传(断点上传),那么断点下载又是什么呢?其实完全可以按照上传的理解
来理解断点续传、分片下载。下载文件的时候将一个大文件分成N个部分进行下载,然后前端再进行组合。
最终得到一个完整的文件。
但是呢,下载跟上传,后端的实现方式还是有区别的,上传需要把接口分成4个接口;但是下载不需要,
一个接口搞定;主要依赖http的Range(关于range,网上资料应该不少)头来进行处理(其实个人还考虑过
另外一种方式,未验证不知道是否可行;方式就是后端将文件进行切割,然后提供一个接口告诉前端某个文
件有多少个分片,前端分别调用接口获取各个分片,然后将分片文件进行合并,此方式是参考到分片上传的
假想)。此方法同样支持普通下载,不传入Range头就可进行普通下载;也可一次只下载一段(传入一个
range:bytes=0-10240);也可下载多段(传入多个range:bytes=0-10240,10241-20480);也可一次下载完文件
(range范围为整个文件即可:bytes=0-102400);前端怎样配合实现完全不知道,如果有哪位大佬知道的话,
真心求教!下面开始进行代码的编码
实现:
1. 下载接口实现:
/**
* 文件下载
* @author kevin
* @param response :
* @param range :
* @param filePath :
* @date 2021/1/17
*/
@ApiOperation(value = "文件下载", notes = "downloadFile")
@GetMapping(value = "/downloadFile")
public void downloadFile(@RequestParam("fileId") String fileId, @RequestParam(name = "filePath",
required = false) String filePath, HttpServletResponse response,
@RequestHeader(name = "Range", required = false) String range) {
List<FileInfo> fileInfo= fileMapper.getFileById(fileId);
if(null == fileInfo){
throw new RuntimeException("下载失败,未找到需要下载的文件");
}
filePath = StringUtils.isNotBlank(filePath) ? filePath : fileInfo.getFilePath();
File file = new File(filePath);
String filename = file.getName();
long length = file.length();
Range full = new Range(0, length - 1, length);
List<Range> ranges = new ArrayList<>();
//处理Range
try {
if (!file.exists()) {
String msg = "需要下载的文件不存在:" + file.getAbsolutePath();
log.error(msg);
throw new RuntimeException(msg);
}
if (file.isDirectory()) {
String msg = "需要下载的文件的路径对应的是一个文件夹:" + file.getAbsolutePath();
log.error(msg);
throw new RuntimeException(ResponseState.REQUEST_ERROR.getCode(), msg);
}
dealRanges(full, range, ranges, response, length);
}catch (IOException e){
e.printStackTrace();
throw new RuntimeException("文件下载异常:" + e.getMessage());
}
// 如果浏览器支持内容类型,则设置为“内联”,否则将弹出“另存为”对话框. attachment inline
String disposition = "attachment";
// 将需要下载的文件段发送到客服端,准备流.
try (RandomAccessFile input = new RandomAccessFile(file, "r");
ServletOutputStream output = response.getOutputStream()) {
//最后修改时间
FileTime lastModifiedObj = Files.getLastModifiedTime(file.toPath());
long lastModified = LocalDateTime.ofInstant(lastModifiedObj.toInstant(),
ZoneId.of(ZoneId.systemDefault().getId())).toEpochSecond(ZoneOffset.UTC);
//初始化response.
response.reset();
response.setBufferSize(20480);
response.setHeader("Content-type", "application/octet-stream;charset=UTF-8");
response.setHeader("Content-Disposition", disposition + ";filename=" +
URLEncoder.encode(filename, StandardCharsets.UTF_8.name()));
response.setHeader("Accept-Ranges", "bytes");
response.setHeader("ETag", URLEncoder.encode(filename, StandardCharsets.UTF_8.name()));
response.setDateHeader("Last-Modified", lastModified);
response.setDateHeader("Expires", System.currentTimeMillis() + 604800000L);
//输出Range到response
outputRange(response, ranges, input, output, full, length);
output.flush();
response.flushBuffer();
}catch (Exception e){
e.printStackTrace();
throw new RuntimeException("文件下载异常:" + e.getMessage());
}
}
/**
* 处理请求中的Range(多个range或者一个range,每个range范围)
* @author kevin
* @param range :
* @param ranges :
* @param response :
* @param length :
* @date 2021/1/17
*/
private void dealRanges(Range full, String range, List<Range> ranges, HttpServletResponse response,
long length) throws IOException {
if (range != null) {
// Range 头的格式必须为 "bytes=n-n,n-n,n-n...". 如果不是此格式, 返回 416.
if (!range.matches("^bytes=\\d*-\\d*(,\\d*-\\d*)*$")) {
response.setHeader("Content-Range", "bytes */" + length);
response.sendError(HttpServletResponse.SC_REQUESTED_RANGE_NOT_SATISFIABLE);
return;
}
// 处理传入的range的每一段.
for (String part : range.substring(6).split(",")) {
part = part.split("/")[0];
// 对于长度为100的文件,以下示例返回:
// 50-80 (50 to 80), 40- (40 to length=100), -20 (length-20=80 to length=100).
int delimiterIndex = part.indexOf("-");
long start = Range.sublong(part, 0, delimiterIndex);
long end = Range.sublong(part, delimiterIndex + 1, part.length());
//如果未设置起始点,则计算的是最后的 end 个字节;设置起始点为 length-end,结束点为length-1
//如果未设置结束点,或者结束点设置的比总长度大,则设置结束点为length-1
if (start == -1) {
start = length - end;
end = length - 1;
} else if (end == -1 || end > length - 1) {
end = length - 1;
}
// 检查Range范围是否有效。如果无效,则返回416.
if (start > end) {
response.setHeader("Content-Range", "bytes */" + length);
response.sendError(HttpServletResponse.SC_REQUESTED_RANGE_NOT_SATISFIABLE);
return;
}
// 添加Range范围.
ranges.add(new Range(start, end, end - start + 1));
}
}else{
//如果未传入Range,默认下载整个文件
ranges.add(full);
}
}
/**
* output写流输出到response
* @author kevin
* @param response :
* @param ranges :
* @param input :
* @param output :
* @param full :
* @param length :
* @date 2021/1/17
*/
private void outputRange(HttpServletResponse response, List<Range> ranges, RandomAccessFile input,
ServletOutputStream output, Range full, long length) throws IOException {
if (ranges.isEmpty() || ranges.get(0) == full) {
// 返回整个文件.
response.setContentType("application/octet-stream;charset=UTF-8");
response.setHeader("Content-Range", "bytes " + full.start + "-" + full.end + "/" + full.total);
response.setHeader("Content-length", String.valueOf(full.length));
response.setStatus(HttpServletResponse.SC_OK); // 200.
Range.copy(input, output, length, full.start, full.length);
} else if (ranges.size() == 1) {
// 返回文件的一个分段.
Range r = ranges.get(0);
response.setContentType("application/octet-stream;charset=UTF-8");
response.setHeader("Content-Range", "bytes " + r.start + "-" + r.end + "/" + r.total);
response.setHeader("Content-length", String.valueOf(r.length));
response.setStatus(HttpServletResponse.SC_PARTIAL_CONTENT); // 206.
// 复制单个文件分段.
Range.copy(input, output, length, r.start, r.length);
} else {
// 返回文件的多个分段.
response.setContentType("multipart/byteranges; boundary=MULTIPART_BYTERANGES");
response.setStatus(HttpServletResponse.SC_PARTIAL_CONTENT); // 206.
// 复制多个文件分段.
for (Range r : ranges) {
//为每个Range添加MULTIPART边界和标题字段
output.println();
output.println("--MULTIPART_BYTERANGES");
output.println("Content-Type: application/octet-stream;charset=UTF-8");
output.println("Content-length: " + r.length);
output.println("Content-Range: bytes " + r.start + "-" + r.end + "/" + r.total);
// 复制多个需要复制的文件分段当中的一个分段.
Range.copy(input, output, length, r.start, r.length);
}
// 以MULTIPART文件的边界结束.
output.println();
output.println("--MULTIPART_BYTERANGES--");
}
}
2.Range类
private static class Range {
long start;
long end;
long length;
long total;
/**
* Range段构造方法.
*
* @param start range起始位置.
* @param end range结束位置.
* @param total range段的长度.
*/
public Range(long start, long end, long total) {
this.start = start;
this.end = end;
this.length = end - start + 1;
this.total = total;
}
public static long sublong(String value, int beginIndex, int endIndex) {
String substring = value.substring(beginIndex, endIndex);
return (substring.length() > 0) ? Long.parseLong(substring) : -1;
}
private static void copy(RandomAccessFile randomAccessFile, OutputStream output, long fileSize, long start, long length) throws IOException {
byte[] buffer = new byte[4096];
int read = 0;
long transmitted = 0;
if (fileSize == length) {
randomAccessFile.seek(start);
//需要下载的文件长度与文件长度相同,下载整个文件.
while ((transmitted + read) <= length && (read = randomAccessFile.read(buffer)) != -1){
output.write(buffer, 0, read);
transmitted += read;
}
//处理最后不足buff大小的部分
if(transmitted < length){
log.info("最后不足buff大小的部分大小为:" + (length - transmitted));
read = randomAccessFile.read(buffer, 0, (int)(length - transmitted));
output.write(buffer, 0, read);
}
} else {
randomAccessFile.seek(start);
long toRead = length;
//如果需要读取的片段,比单次读取的4096小,则使用读取片段大小读取
if(toRead < buffer.length){
output.write(buffer, 0, randomAccessFile.read(new byte[(int) toRead]));
return;
}
while ((read = randomAccessFile.read(buffer)) > 0) {
toRead -= read;
if (toRead > 0) {
output.write(buffer, 0, read);
} else {
output.write(buffer, 0, (int) toRead + read);
break;
}
}
}
}
@Override
public boolean equals(Object o) {
if (this == o) return true;
if (o == null || getClass() != o.getClass()) return false;
Range range = (Range) o;
return start == range.start &&
end == range.end &&
length == range.length &&
total == range.total;
}
@Override
public int hashCode() {
final int prime = 31;
return Objects.hash(prime, start, end, length, total);
}
}