使用客户端难免会碰到这样的场景,用户需要缓存一个电影,他点击下载按钮,然后将应用推到后台,过了一两个小时,打开app看到电影已经下完了。 实现这个需求有两种做法
后台下载
用后台任务,当app退到后台后,过一段时间应用会挂起,但是依然不影响下载。并且下载完成后,会进入相应的回调方法中。应用保活
参照腾讯视频这样的应用,推到后台,会申请backgroundaudio权限,然后播放一段无声的音乐,这样应用就和在前台一样,然后下载任务完成时,就把后台保活退出。-
断点续传
文件下载过程中,文件会存在temp下。等到下载完成,会通过相应的代理方法告诉delegate 还有一个问题,如果电影下载到一半,用户取消了,然后又重新下,这时应该从断掉的地方开始下载才对,这个问题容易解决,Apple 文档里有一个方法
[self.task cancelByProducingResumeData:^(NSData * _Nullable resumeData) {
[resumeData writeToFile:[weakSelf getTmpFileUrl] atomically:NO];
NSLog(@"hellozmodo%s----%@",__func__,resumeData);
}];
但是如果正在下载,用户kill掉了app,那就有点麻烦,断点下载需要一个NSdata类型的resumeData,它其实是一个xml文件。里面信息包括了下载URL、已接收字节数、临时的下载文件名(文件默认存在tmp文件夹中)、当前请求、原始请求、下载事件、resumeInfo版本、EntityTag这些数据
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
<plist version="1.0">
<dict>
<key>NSURLSessionDownloadURL</key>
<string>http://downloadUrl</string>
<key>NSURLSessionResumeBytesReceived</key>
<integer>1474327</integer>
<key>NSURLSessionResumeCurrentRequest</key>
<data>
......
</data>
<key>NSURLSessionResumeEntityTag</key>
<string>"XXXXXXXXXX"</string>
<key>NSURLSessionResumeInfoTempFileName</key>
<string>CFNetworkDownload_XXXXX.tmp</string>
<key>NSURLSessionResumeInfoVersion</key>
<integer>2</integer>
<key>NSURLSessionResumeOriginalRequest</key>
<data>
.....
</data>
<key>NSURLSessionResumeServerDownloadDate</key>
<string>week, dd MM yyyy hh:mm:ss </string>
</dict></plist>
所以我们可以考虑自己合成一个resumeData,通过调试,查看信息,发现NSURLSessionDownloadTask中有个数据downloadFile存放了一些关于下载的信息,其中一个信息path就是存放临时文件路径的,通过lastPathComponent就可以直接取到相应的临时文件名。通过tmp文件名获取tmp文件路径,这样做是因为本地文件路径会变,所以不能直接存task中的文件路径,需要获取到文件名,通过tmp的路径获取到tmp文件路径。