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

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

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

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

  • 断点续传


    大文件后台下载以及断点下载最优解决方案_第1张图片
    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这些数据





     NSURLSessionDownloadURL
         http://downloadUrl
     NSURLSessionResumeBytesReceived
         1474327
     NSURLSessionResumeCurrentRequest
         
          ......
         
     NSURLSessionResumeEntityTag
         "XXXXXXXXXX"
     NSURLSessionResumeInfoTempFileName
         CFNetworkDownload_XXXXX.tmp
     NSURLSessionResumeInfoVersion
         2
     NSURLSessionResumeOriginalRequest
         
          .....
         
     NSURLSessionResumeServerDownloadDate
          week, dd MM yyyy hh:mm:ss 

所以我们可以考虑自己合成一个resumeData,通过调试,查看信息,发现NSURLSessionDownloadTask中有个数据downloadFile存放了一些关于下载的信息,其中一个信息path就是存放临时文件路径的,通过lastPathComponent就可以直接取到相应的临时文件名。通过tmp文件名获取tmp文件路径,这样做是因为本地文件路径会变,所以不能直接存task中的文件路径,需要获取到文件名,通过tmp的路径获取到tmp文件路径。

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

你可能感兴趣的:(大文件后台下载以及断点下载最优解决方案)