http请求中的range和Content-Range

http请求中的range和Content-Range,第1张

一、问题的由来

关注这个问题,是因为最近做了一个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("-","") +"\"")

```

range的值当然可以在前端设置好了,然后调用后台的setHeader进行设置range的值。

uriRequest. setHeader ("Range", "bytes=" + current + "-")


欢迎分享,转载请注明来源:内存溢出

原文地址: http://outofmemory.cn/bake/11518364.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2023-05-16
下一篇 2023-05-16

发表评论

登录后才能评论

评论列表(0条)

保存