最近更新时间:2023.12.20 16:08:35
首次发布时间:2022.04.22 11:09:16
本文介绍如何通过 iOS 点播 SDK 搭建“抖音”同款短视频场景,实现“零首帧”的短视频播放效果。
在短视频场景中,缩短视频首帧耗时和提升视频上下滑动的流畅度是提升用户观看体验的重要手段:
视频首帧耗时是指从用户点击开始播放按钮到视频首帧画面展现出来的时间。视频首帧耗时是视频类应用的核心指标之一,直接影响用户的观看体验。如果首帧加载需要几秒钟,用户可能会感到不耐烦,甚至选择放弃观看。短视频场景致力于实现“零首帧”,并不是真正的 0 毫秒起播,而是用户几乎感知不到有首帧耗时的存在。
在短视频场景中,用户通常通过上下滑动来浏览不同的视频内容。如果视频在滑动过程中出现卡顿,会阻碍用户观看。提升视频上下滑动的流畅度可以确保用户更为轻松地浏览视频和享受视频内容,提高整体观看流畅度和连贯性,从而增加人均播放量和用户停留时长。
为了帮助您快速搭建“抖音”同款短视频场景,点播 SDK 基于抖音亿级日活跃用户的真实反馈和大规模实践经验,提供两大最佳策略:预加载策略和预渲染策略。此外,您还可参考示例 Demo,实现播放提前、播放器异步调用和设置封面图等策略。这些策略可单独应用,也可相互搭配。
通过采用这些策略,您可将视频首帧耗时缩短至 100-300 毫秒,为用户呈现极速首帧和流畅丝滑的播放体验,让用户享受到与抖音相媲美的视觉盛宴。
以下视频展示了接入短视频场景最佳实践前后的播放效果对比。
接入短视频场景最佳实践前 | 接入短视频场景最佳实践后 |
---|---|
上下滑动视频时有明显的卡顿感 | 播放流畅丝滑 |
下载地址:iOS 示例项目。
策略 | 说明 | 优化效果 |
---|---|---|
预加载策略 | 预加载是指在播放当前视频时,提前下载后续视频的数据,以便在切换到下一个视频时实现快速起播,从而优化用户的播放体验。 | 缩短视频首帧耗时 |
预渲染策略 | 预渲染是指在播放当前视频时,提前创建播放器,对下一个视频进行解码和渲染操作。同时,您还可将预渲染的首帧用作视频封面,提前展示给用户,增强视觉效果。 | 缩短视频首帧耗时 |
播放提前 | 在上下滑动视频时,建议您在用户手指抬起后、下一个页面将要显示时调用播放接口,而非等待滑动停止后再调用播放接口。具体实现可参考 iOS 示例项目。 | 缩短视频首帧耗时 |
播放器异步调用 | 建议您使用点播 SDK 提供的异步接口(例如 | 提升视频上下滑动的流畅度 |
设置封面图 | 建议您使用视频默认抽取的非黑首帧作为封面图,以避免在画面尚未渲染时出现黑屏无内容的情况。如果您不自行设置封面图,点播 SDK 的预渲染策略也会抽取首帧作为封面图,但该策略仅适用于向下滑动的视频,并且不能预渲染过多的视频,以避免性能消耗。因此,建议您选择自行设置封面图以提升用户体验。 | 避免出现黑屏,提升用户体验 |
为接入预加载策略或预渲染策略,您需要使用以下方式设置播放源。
参考以下示例代码设置 Vid 播放源。除了 vid
和 playAuthToken
外,您还需指定 resolution
,且此处设置的 resolution
必须和设置预加载源时的 resolution
保持一致。
/// 创建播放源 NSString *vid = @"YOUR_VIDEO_ID"; // appServer 下发 NSString *playAuthToken = @"YOUR_PLAYAUTHTOKEN"; // appServer 下发 TTVideoEngineResolutionType resolution = TTVideoEngineResolutionTypeHD; TTVideoEngineVidSource *source = [[TTVideoEngineVidSource alloc] initWithVid:vid playAuthToken:playAuthToken resolution:resolution]; /// 设置播放源 [engine setVideoEngineVideoSource:source]; /// 播放 [engine play];
参考以下示例代码设置 DirectUrl 播放源。url
、vid
和 cacheKey
均为必填参数。其中:
vid
:视频 ID,可以使用 Vid 或者 Vid 的 MD5 值。
cacheKey
:缓存 key,即 URL 的唯一标识。每个视频可能存在多个不同清晰度的 URL,每个 URL 对应一个 cacheKey
,但是多个 URL 对应同一个视频 ID。您可以使用 URL 中不变的部分进行 MD5 操作,将 MD5 值用作 cacheKey
。例如,对于 https://www.xxx.com/xxx/x/number
, number
之前的部分是不变的。
/// 创建播放源 NSString *vid = @"YOUR_VIDEO_ID"; // 替换成您自己的视频 ID NSString *url = @"http://www.example.com/test.mp4";// 替换成您自己的视频 URL NSString *cacheKey = [url vloc_md5String]; TTVideoEngineUrlSource *source = [[TTVideoEngineUrlSource alloc] initWithUrl:url cacheKey:cacheKey videoId:vid]; /// 设置播放源 [engine setVideoEngineVideoSource:source]; /// 播放 [engine play];
为接入预加载策略,您需要将视频播放列表传入 SDK 作为预加载源。SDK 会根据您提供的播放列表,在播放当前视频时提前下载后续的视频内容。在此过程中,SDK 会根据当前的播放状态自动管理预加载的时机和配置,例如预加载的并发数和数据大小。
接入预加载策略只需简单的几行代码即可完成。以下是具体的步骤:
在用户进入短视频页面时,调用 enableEngineStrategy:scene: 开启预加载策略。示例代码如下:
/// 开启预加载策略,建议在进入短视频页面开启 [TTVideoEngine enableEngineStrategy:TTVideoEngineStrategyTypePreload scene:TTVEngineStrategySceneSmallVideo];
在适当的时机设置预加载源:
在首次加载播放源数据或在刷新页面以显示新数据的场景中,调用 setStrategyVideoSources:
在加载更多数据的场景中,调用 addStrategyVideoSources:
示例代码如下:
- (void)configVideoEngineStrategyMediaSources:(NSArray<VEVideoModel *> *)videoModels refresh:(BOOL)refresh { NSMutableArray *sources = [NSMutableArray array]; [videoModels enumerateObjectsUsingBlock:^(VEVideoModel * _Nonnull obj, NSUInteger idx, BOOL * _Nonnull stop) { [sources addObject:[VEVideoModel videoEngineVidSource:obj]]; }]; if (refresh) { /// 首次加载或者下拉刷新 [TTVideoEngine setStrategyVideoSources:sources]; } else { /// 上拉加载更多 [TTVideoEngine addStrategyVideoSources:sources]; } }
注意
错误的播放列表会导致预加载失败。
预加载策略会查询到当前播放视频在播放列表中的 index,并预加载 index+1、index+2 ... index+n 的视频。
如果您需要自定义预加载机制,可通过 SDK 提供的预加载接口自行实现,详见自定义预加载。
为接入预渲染策略,您需要将视频播放列表传入 SDK 作为预渲染源。SDK 会根据您提供的播放列表,在播放当前视频时,提前创建播放器,对播放列表中的下一个视频进行解码和渲染操作。
注意
点播 SDK 当前只支持预渲染下一个视频,不支持预渲染上一个视频。
接入预渲染策略的具体步骤如下:
在用户进入短视频页面时,调用 enableEngineStrategy:scene: 开启预渲染策略。示例代码如下:
/// 开启预加载策略,建议在进入短视频页面开启 [TTVideoEngine enableEngineStrategy:TTVideoEngineStrategyTypePreRender scene:TTVEngineStrategySceneSmallVideo];
在适当的时机设置预渲染源:
在首次加载播放源数据或在刷新页面以显示新数据的场景中,调用 setStrategyVideoSources:
在加载更多数据的场景中,调用 addStrategyVideoSources:
示例代码如下:
- (void)configVideoEngineStrategyMediaSources:(NSArray<VEVideoModel *> *)videoModels refresh:(BOOL)refresh { NSMutableArray *sources = [NSMutableArray array]; [videoModels enumerateObjectsUsingBlock:^(VEVideoModel * _Nonnull obj, NSUInteger idx, BOOL * _Nonnull stop) { [sources addObject:[VEVideoModel videoEngineVidSource:obj]]; }]; if (refresh) { /// 首次加载或者下拉刷新 [TTVideoEngine setStrategyVideoSources:sources]; } else { /// 上拉加载更多 [TTVideoEngine addStrategyVideoSources:sources]; } }
注意
预渲染策略会查询到当前播放视频在播放列表中的 index,预渲染 index+1 视频源。
仅支持预渲染 1 个视频。
(可选)调用 setPreRenderVideoEngineDelegate: 设置预渲染策略回调,再根据自身业务需要对 Engine 实例进行其它设置,例如设置视频填充模式、起播时间等。
/// 设置预渲染策略回调 + (void)setPreRenderVideoEngineDelegate:(id<TTVideoEnginePreRenderDelegate>)delegate;
调用 getPreRenderVideoEngineWithVideoSource: 判断预渲染是否完成,然后处理播放逻辑。示例代码如下:
/// 通过播放视频 Source 查询是否已经预渲染完成 TTVideoEngine *preRenderVideoEngine = [TTVideoEngine getPreRenderVideoEngineWithVideoSource:mediaSource]; if (preRenderVideoEngine) { /// 预渲染完成,直接调用播放即可 [preRenderVideoEngine play]; } else { /// 预渲染未完成,创建 TTVideoEngine 播放 }
在退出短视频页面或切换到其他页面时调用 clearAllEngineStrategy 关闭所有策略,释放资源。
- (void)close { /// 务必调用 clear 方法 [TTVideoEngine clearAllEngineStrategy]; }
如果您同时接入预加载策略和预渲染策略,仅需设置一次视频播放列表。