我正在创建一个 Java 应用程序,它通过 http 将视频文件“流式传输”到浏览器(当前为 Chrome v24.x)。该视频被发送到 FFmpeg,并且其输出通过 HTTP 发送。
现在,一旦文件被完全编码,文件就会使用分块传输提供服务,并响应范围请求。示例标题:
要求
GET /file/9fe6b502-c127-47c2-b6d2-83ea58676a8d HTTP/1.1 : Host: localhost:1234 : Connection: keep-alive : Accept-Encoding: identity;q=1, *;q=0 : User-Agent: Mozilla/5.0 (Macintosh; Intel Mac OS X 10_8_2) AppleWebKit/537.17 (KHTML, like Gecko) Chrome/24.0.1312.56 Safari/537.17 : Accept: */* : Referer: http://localhost:1234/media/9fe6b502-c127-47c2-b6d2-83ea58676a8d : Accept-Language: en-US,en;q=0.8 : Accept-Charset: ISO-8859-1,utf-8;q=0.7,*;q=0.3 : Cookie: plushContainerWidth=100%25; plushNoTopMenu=0 : Range: bytes=0- :
回复
HTTP/1.1 206 Partial Content : Connection: close : Date: Mon, 28 Jan 2013 14:51:52 GMT : Content-Type: video/mp4 : Etag: "9fe6b502-c127-47c2-b6d2-83ea58676a8d" : Accept-Ranges: bytes : Content-Range: bytes 0-625825/625826 : Content-Length: 625826 : Transfer-Encoding: chunked :
发送的数据为 625826 字节,不包括标头数据和分块开销。
现在,这工作得很好!
问题在于 GET 请求发生在文件编码完成之前。我试图通过 HTTP 立即开始发送文件,只使用没有内容长度属性或范围的分块传输,因为它们目前不知道。这会导致浏览器等待完整的文件(直到传输完成才开始播放)。另外,文件传输完成后,浏览器报视频错误,无法播放文件。示例标题:
要求
Request : GET /file/9fe6b502-c127-47c2-b6d2-83ea58676a8d HTTP/1.1 : Host: localhost:1234 : Connection: keep-alive : Accept-Encoding: identity;q=1, *;q=0 : User-Agent: Mozilla/5.0 (Macintosh; Intel Mac OS X 10_8_2) AppleWebKit/537.17 (KHTML, like Gecko) Chrome/24.0.1312.56 Safari/537.17 : Accept: */* : Referer: http://localhost:1234/media/9fe6b502-c127-47c2-b6d2-83ea58676a8d : Accept-Language: en-US,en;q=0.8 : Accept-Charset: ISO-8859-1,utf-8;q=0.7,*;q=0.3 : Cookie: plushContainerWidth=100%25; plushNoTopMenu=0 : Range: bytes=0-
回复
HTTP/1.1 200 OK : Connection: close : Date: Mon, 28 Jan 2013 14:51:29 GMT : Content-Type: video/mp4 : Transfer-Encoding: chunked
发送的数据为 625826 字节,不包括标头数据和分块开销。
有谁知道出了什么问题,或者如何在不知道文件全长的情况下开始播放视频?
谢谢你的时间,
詹姆士。
//编辑
由于对不完整文件的请求状态为“范围:字节 = 0-” - 我可以使用 Content-Range:字节 0-999/* 回复部分内容为“n”字节(例如,1000 字节)吗?
//EDIT 2 根据要求,这是我输出文件的代码。这是浓缩的,因为代码实际上跨越了几个类。
File f = new File(_filename);
RandomAccessFile raf = new RandomAccessFile(f, "r");
ChunkedOutputStream cos = new ChunkedOutputStream(_out, 1024 * 100);
byte[] bytes;
while (true){
lBufferSizeMax = Math.min(lBufferSize, fc.length() - lCompleted);
bytes = new byte[(int)lBufferSizeMax];
lCurrentRead = fc.read(bytes);
if (lCurrentRead == 0){
break;
}
cos.write(bytes);
}