我假设您询问的是 HTTP Live Streaming 视频,如您的示例 URL 所示,而不是 Smooth Streaming 视频。如果不是这种情况,请发表评论,我将编辑答案以谈论平滑流。
HTTP 直播视频的结构
HTTP Live Streaming (HLS) 有多个版本,其中较新的版本添加了对多语言音频和字幕的适当支持,这使场景变得非常复杂。我会假设您对这些功能不感兴趣,并会专注于简单的案例。
HLS 具有三层结构:
- 在根目录下,您拥有 主播放列表。这是您请求视频根 URL 时 Web 服务器提供的内容。它包含对一个或多个媒体播放列表的引用。
-
媒体播放列表代表一个特定配置的整个视频。例如,如果媒体使用两个质量级别(例如 720p 和 1080p)进行编码,则将有两个媒体播放列表,每个播放列表一个。媒体播放列表包含对实际包含媒体数据的媒体片段的引用列表。
-
媒体片段是 MPEG 传输流,其中包含一段数据流,通常每个文件大约 10 秒。
如果不考虑多语言功能,可以将 HLS 视频视为分成 10 秒块的多个独立视频 - 所有视频都包含相同的内容但使用不同的质量级别。
上述每个实体——主播放列表、媒体播放列表、每个媒体片段——都由播放器使用标准 HTTP 文件下载机制单独下载。
将碎片重新组合在一起
媒体播放器所需的所有信息都存在于媒体片段中 - 您几乎可以忽略主播放列表和媒体播放列表,因为它们的唯一目的是为您提供媒体片段的 URL。
幸运的是,MPEG 传输流格式本质上非常简单。为了将媒体片段重新组合在一起,您需要做的就是将它们连接在一起。就是这样,真的!
伪代码
我假设您不是在询问如何使用 Objective-C 执行 HTTP 请求,因为 Stack Overflow 上关于该主题还有许多其他答案。相反,我将重点介绍您需要实现的算法。
首先,您只需下载主播放列表。
masterPlaylist = download(rootUrl);
主播放列表包含注释行和数据行。每条数据线都是对媒体播放列表的引用。请注意,HLS 的最低质量级别通常只有音频流。为简单起见,我们假设您关心文件中的第一质量级别。
masterPlaylistLines = masterPlaylist.split('\n');
masterPlaylistDataLines = filter(masterPlaylistLines, x => !x.startsWith("#"));
firstMasterPlaylistDataLine = masterPlaylistDataLines[0];
此数据行将包含媒体播放列表的相对 URL。让我们下载它。 URL 附加代码应该很聪明,并且理解如何制作相对 URL,而不仅仅是字符串连接。
mediaPlaylist = download(rootUrl + firstMasterPlaylistDataLine);
反过来,媒体播放列表的格式相同,但包含对媒体片段的引用。让我们将它们全部下载并附加到一起。
mediaPlaylistLines = mediaPlaylist.split('\n');
mediaPlaylistDataLines = filter(mediaPlaylistLines, x => !x.startsWith("#"));
foreach (dataLine : mediaPlaylistDataLines)
{
// URL concatenation code is assumed to be smart, not just string concatenation.
mediaSegment = download(rootUrl + firstMasterPlaylistDataLine + dataLine);
appendToFile("Output.ts", mediaSegment);
}
最终输出将是单个 MPEG 传输流文件,可在大多数现代媒体播放器上播放。如果您想将其转换为另一种格式,可以使用各种免费工具,例如 FFmpeg。