一、问题的由来
关注这个问题,是因为最近做了一个spring boot服务,用后台逻辑代理了一个mp4视频的请求, 其实就是用java读取mp4的内容,再写入到HttpServletResponse中。
用pc浏览器都无问题, 但发现用手机移动端浏览器(iphone)来浏览时,无法播放,通过实时的调试发现,iphone的浏览器在请求 mp4播放时,会先发送一个带range头的http连接,先请求视频的第0~1个字节,然后再使用range来分段的请求mp4数据的;首先发送一个很小的range请求,估计是为了先探知文件的大小,后面再分段请求。而我的后台逻辑没有兼顾range请求,每次文件内容都全部发送,而在服务端写入reponse时会看到一个piped error。其实就是客户端觉得你这个服务器端不按套路出牌,关闭了链接、不再发起后续的分片请求。
二、相关的range 参数、格式
1、客户端在发送range请求,会在请求头中增加range请求头
常用格式为: Range: bytes=first-end
first,开始数据的索引位置
end,结束数据的索引位置
例如
Range: bytes=0-499 其实就是前500个字符
Range: bytes=2-10 第3个字符(索引位置为2)~第11个字符(索引位置为10)
Range: bytes=0- 如省略第二个参数,即从索引开始位置,到结束位置
Range:bytes=-500 如省略地一个参数,即表示最后500个字符
多个集合模式, Range: bytes=p1-p2,m1-m2 表示多个区段,用","分割
2、服务器回应格式 Content-Range: bytes first-end/total 以及 Content-Length: len
在Content-Range中:
first,数据的开始数据的位置索引 (inclusive)
end, 数据的结束位置索引(inclusive)
total, 数据的整包大小
在Content-Length中
len: 此次回应的数据大小, 注意该字段和total并不一定相等,是此次请求回应的总大小,其实就是 end-first + 1
3、状态码
在收到Range请求后, 如服务器支持断点续传,可回应 206、否则200; 如果请求的range错误,回应 416
三、一个接收range请求,分拆数据、判定合法的小例子(没考虑多段range)
```java
long entityLength =file.length();
long startPos = -1, endPos = -1;
String rangeHeader = request.getHeader("range");
if (rangeHeader !=null &&rangeHeader.trim() !="" &&rangeHeader.startsWith("bytes=")) {
String rangeString =rangeHeader.replaceAll("bytes=","");
if(rangeString.startsWith("-")) {
endPos =entityLength -1;
startPos = endPos -Integer.parseInt(new String(rangeString.getBytes(),1,rangeString.length() -1)) +1;
}
else if(rangeString.endsWith("-")) {
endPos =entityLength -1;
startPos =Integer.parseInt(new String(rangeString.getBytes(),0,rangeString.length() -1));
}
else {
String[]rangePs =rangeString.split("-");
if(rangePs.length ==2) {
startPos =Integer.parseInt(rangePs[0]);
endPos =Integer.parseInt(rangePs[1]);
}
}
if(startPos <0 || endPos <0 || startPos >=entityLength || endPos >=entityLength || startPos > endPos) {
response.setStatus(HttpServletResponse.SC_REQUESTED_RANGE_NOT_SATISFIABLE);
return;
}
response.setStatus(HttpServletResponse.SC_PARTIAL_CONTENT);
}
long contentLength = endPos - startPos +1;
response.addHeader("Content-Type","video/mp4");
response.addHeader("Connection","keep-alive");
response.setHeader("Content-Range",String.format("bytes %d-%d/%d", startPos, endPos,entityLength));
response.addHeader("Content-Length",String.valueOf(contentLength));
response.addHeader("ETag","\"" +task.getTaskId().replaceAll("-","") +"\"");
```