我正在创建一个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-" - 我可以用内容范围:字节 0-999/* 回复"n"字节(例如 1000 字节)的部分内容吗?
编辑 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);
}
,当编码结束时,ffmpeg
可能会重写文件的某些部分。 这意味着视频文件中的元数据直到最后都是不完整的,因此尝试在该时间之前流式传输文件的开头是没有用的。 您可以检查将ffmpeg
命令行中的输出文件名替换为单个破折号,并将标准输出重定向到文件。 而不是ffmpeg ... out.mov
做ffmpeg ... - > out.mov
. 如果您收到一条错误消息,指出格式需要可搜索的输出,则会受到影响。
我已经完成了几乎完全符合您要做的事情,结果在我的 Git 存储库中。 我使用的容器格式是 FLV,我可以在其中使用元数据玩游戏。 不幸的是,这不是iPad/iPhone设备喜欢看到的。