这段时间工作不忙,把视频离线缓存重新研究了一下,以前也是研究过。参考了一些其他文章和资料,文章链接找不到了,敬请谅解。
把逻辑代码封装了一下,支持视频的离线缓存、边下边播、快进、断网重连、数据自动保存到本地、当下次重新播放时会优先使用本地数据等。项目 TTPlayerCache 支持CocoaPods,用起来比较方便(支持的视频格式MP4 ,其他没有测试)
demo有注释
1、用法:
pod 'TTPlayerCache'
#import <TTPlayerCache.h>
...
//把视频播放地址转成系统不能识别的URL
NSString *videoUrl = @"http://....";
videoUrl = TTResourceUrlFromOrigianllUrl(videoUrl);
...
...
//设置AVPlayer播放
//初始化代理
self.resourceLoaderDelegate = [TTResourceLoaderDelegate new];
self.urlAsset = [AVURLAsset assetWithURL:self.videoURL];
[self.urlAsset.resourceLoader setDelegate:self.resourceLoaderDelegate queue:TT_resourceLoader_delegate_queue()];
...
就这样。
2、视频播放过程
通过对系统播放器抓包分析(MP4视频格式):
- AVPlayer每次播放时第一次都会先请求bytes=0-2的数据,获得到视频的总字节数、视频类型等信息
- 第二次请求全部数据bytes=x-,当数据响应填充后可能会有不同的反应,请求全部数据是requestsAllDataToEndOfResource == YES。
- 当视频快进时,会取消先前的下载任务,如果快进区域没有缓冲也会调用resourceLoader: shouldWaitForLoadingOfRequestedResource:方法,这是会出现requestsAllDataToEndOfResource == NO/YES的loadingRequest ,然后继续重复。
...
当视频缓存到一定程度时系统会调用resourceLoader: didCancelLoadingRequest:取消下载任务。
3、代码分析:
代码比较少,主要是TTResourceLoaderDelegate和TTResourceLoaderData这两个类
- TTResourceLoaderDelegate 实现AVAssetResourceLoaderDelegate的两个代理方法。
- (BOOL)resourceLoader:(AVAssetResourceLoader *)resourceLoader shouldWaitForLoadingOfRequestedResource:(AVAssetResourceLoadingRequest *)loadingRequest {
//对loadingRequest进行处理,数据回填
...
return YES;
}
- (void)resourceLoader:(AVAssetResourceLoader *)resourceLoader didCancelLoadingRequest:(AVAssetResourceLoadingRequest *)loadingRequest {
//取消请求处理
...
}
- TTResourceLoaderData实现对loadingRequest响应数据、缓存、判断是否继续请求数据(还是使用本地数据)、请求任务取消等操作。
- TTResourceLoaderData对下载数据的处理类。
- TTResourceLoaderCache对本地缓存数据的处理,可以获取总缓存大小、清空全部缓存和删除某个缓存。
- TTPlayerCacheMacro定义了TTPlayerCache的Scheme等常量、URL转换方法。
- TTReachability 是对Reachability源码加了前缀方便使用,管理视频播放过程中断网处理。
- (void)handleAssetResourceLoadingRequest:(AVAssetResourceLoadingRequest *)loadingRequest {...}
实现对loadingRequest的处理,根据loadingRequest.dataRequest.requestsAllDataToEndOfResource == YES与否分别处理(是否等待请求全部数据),封装不同的NSURLRequest,判断是否使用本地数据。requestsAllDataToEndOfResource 为YES的下载任务只有一个,在开始之前取消上一个等等。
下载任务收到响应时
- (void)TT_downloadTaskDataTask:(NSURLSessionDataTask *)dataTask didReceiveResponse:(NSURLResponse *)response
,先初始化一个TTResourceLoaderData保存下视频的总字节数和类型,TTResourceLoaderData会创建一个和视频字节数一样长度的NSMutableData用于填充数据。下载任务收到数据时
- (void)TT_downloadTaskDataTask:(NSURLSessionDataTask *)dataTask didReceiveData:(NSData *)data
->调用TTResourceLoaderData- (void)appendData:(NSData *)data taskId:(NSUInteger)taskIdentifier
方法对数据进行处理:
把data放到_data
的正确位置、调整_receivedDataPointArray
(存储已经下载的数据的位置)、loadingRequest.dataReqeust
响应数据 ,具体看代码。下载任务完成或出错
- (void)TT_downloadTaskTask:(NSURLSessionTask *)task didCompleteWithError:(NSError *)error
->调用TTResourceLoaderData- (void)taskCompleteWithError:(NSError *)error taskId:(NSUInteger)taskIdentifier
对下载任务是否取消、网络错误、正常完成分别处理。-
网络恢复时调用
- (void)reloadLoadingRequestWhenHasNetError
刷新播放器。
对下载数据的处理都在TTResourceLoaderData类里完成。重点:
loadingRequest.dataRequest.requestsAllDataToEndOfResource == YES or NO
的处理
TTResourceLoaderData
对象对视频数据实际位置的表示...
第一次写,不对的地方欢迎指正,谢谢!
有问题请留言