大文件后台下载以及断点下载最优解决方案

使用客户端难免会碰到这样的场景,用户需要缓存一个电影,他点击下载按钮,然后将应用推到后台,过了一两个小时,打开app看到电影已经下完了。 实现这个需求有两种做法

  • 后台下载
    用后台任务,当app退到后台后,过一段时间应用会挂起,但是依然不影响下载。并且下载完成后,会进入相应的回调方法中。

  • 应用保活
    参照腾讯视频这样的应用,推到后台,会申请backgroundaudio权限,然后播放一段无声的音乐,这样应用就和在前台一样,然后下载任务完成时,就把后台保活退出。

  • 断点续传


    WechatIMG1.jpeg

    文件下载过程中,文件会存在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文件路径。

参考:https://www.jianshu.com/p/af6700ff91e5
demo

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。