文件下载总结

之前做了文件下载相关的模块,一直忙其他事情,现在终于有时间回顾一下了。文件下载其实分两大块:一是文件本身的下载,二是本地数据关系的维护。

先说文件下载

首先文件下载的需求是前台完整下载、断点下载、后台下载和杀死APP后下载任务保留。需求明确之后回顾一下网络知识。

  • NSURLSession: iOS7以后取代之前的NSURLConnection。有两种获取方式。 一种是获取系统提供的单例,这种方式可以共用NSURLCache, NSHTTPCookieStorage 和 NSURLCredentialStorage。第二种方式是自己创建,自己创建的话
+ (NSURLSession *)sessionWithConfiguration:(NSURLSessionConfiguration *)configuration;
+ (NSURLSession *)sessionWithConfiguration:(NSURLSessionConfiguration *)configuration delegate:(nullable id )delegate delegateQueue:(nullable NSOperationQueue *)queue;

自己创建的话必传参数需要一个配置对象,也可以另外指定代理和代理队列。

  • NSURLSessionConfiguration: 配置管理类,其中封装有超时时长,以及对于是否允许蜂窝数据, 是否允许缓存,是否设置cookie等属性, 系统提供了三种配置管理对象。 defaultSessionConfiguration、ephemeralSessionConfiguration和backgroundSessionConfiguration。 前两种是通过单例获取, 后台管理对象则是根据标识符创建,同样。defaultSessionConfiguration默认的配置管理使用全局单例凭据、缓存和cookie存储对象。ephemeralSessionConfiguration短暂的配置管理对象没有永久的磁盘缓存和cookie存储。backgroundSessionConfiguration后台配置管理对象可在一定约束下对刮起的应用进行网络操作。

  • NSURLSessionTask: 代表一个网络操作或者处理过程, 可以取消(其内部封装有关于一个网络操作的信息, 比如当前状态,进度,是否出错,原始请求,当前请求等)。 为便于区分其有四个子类。分别是NSURLSessionDataTask、NSURLSessionUploadTask、NSURLSessionDownloadTask和NSURLSessionStreamTask(iOS9.0加的)。 其中几个子类和父类并没有本质上的区别, 只是文字意义上加以区分, 分别代表不同的任务。

  • NSURLSessionDelegate: NSURLSession的代理协议, 一共三个个方法, 用于表示session出错, session接收到授权确认,和 有后台数据任务完成的代理

  • NSURLSessionTaskDelegate: 与特定任务的操作相关的消息。该协议遵守NSURLSessionDelegate协议, 其中定义的代理方法包括(即将重定向、收到信任挑战,接收到数据以及出错等)

  • NSURLSessionDataDelegate : 数据task的代理协议,该协议遵守task代理协议, 并且提供额外的代理方法,包含收到响应、dataTask转变为其他task,收到数据等方法

  • NSURLSessionDownloadDelegate : 下载task代理协议,其中定义了下载完成和下载进度改变的代理方法

需求功能的实现
  • 前台文件下载
    经测试最好使用系统提供的NSURLSessionDownloadTask。
  • 断点续传
    NSURLSessionDownloadTask有相应的API, 可以暂停的时候保存resumeData。 然后继续下载的时候根据上次保存的resumeData创建新的NSURLSessionDownloadTask然后继续下载。
  • 后台下载
    使用后台下载时,需要创建对应的后台模式的NSURLSessionConfiguration。
    应用在后台时,如果有任务下载完成或者开启, 会调用APPDelegate的代理方法
    (对于配置为后台模式的NSURLSession, 应用在后台的时候,如果出现任务开启,完成或者授权等活动的时候,为了通知应用会调用APPDelegate的代理方法, 该代理方法有两个参数, 一个时后台session对应的identifier和一个完成的会调block)
- (void)application:(UIApplication *)application handleEventsForBackgroundURLSession:(NSString *)identifier completionHandler:(void (^)(void))completionHandler

代理方法注释说,如果对应identifier的session还没有创建,那么根据identifier创建session好之后会调用相应的代理方法,如果对应identifier的session已经存在,那么会自动调用对应的代理方法(比如下载完成、下载出错、下载开始。最有会调用一个NSURLSession的代理方法如下),不需要做其他操作。还有在做完操作之后要及时的调用回调block。

- (void)URLSessionDidFinishEventsForBackgroundURLSession:(NSURLSession *)session

当走到上面这个代理方法的时候, 上面的开始下载、进度改变和完成出错的方法已经走过了, 也就是说该处理的都已经处理了, 所以一个时候需要调用之前保存的回调block了。

  • 下载中被杀死之后的处理
    对于配置后后台模式的session,如果有任务正在下载的时候杀死应用(杀死应用的时候会把任务给取消), 那么重新启动的时候会调用出错的代理方法
- (void)URLSession:(NSURLSession *)session task:(NSURLSessionTask *)task didCompleteWithError:(NSError *)error

并且error中userInfo中会多一个key 。 NSURLErrorBackgroundTaskCancelledReasonKey还有已经下载的data。根据据此判断是否需要重新添加下载,这样就可以实现应用被杀死之后重新启动的时候继续下载的功能了

下载文件功能完成后,就是本地数据关系的维护了

  • 沙盒知识回顾
    Documents 目录:您应该将所有的应用程序数据文件写入到这个目录下。这个目录用于存储用户数据或其它应该定期备份的信息。
    AppName.app 目录:这是应用程序的程序包目录,包含应用程序的本身。由于应用程序必须经过签名,所以您在运行时不能对这个目录中的内容进行修改,否则可能会使应用程序无法启动。
    Library 目录:这个目录下有两个子目录:Caches 和 Preferences
    Preferences 目录包含应用程序的偏好设置文件。您不应该直接创建偏好设置文件,而是应该使用NSUserDefaults类来取得和设置应用程序的偏好
    Caches 目录用于存放应用程序专用的支持文件,保存应用程序再次启动过程中需要的信息。
    tmp 目录:这个目录用于存放临时文件,保存应用程序再次启动过程中不需要的信息。

  • 本地数据库的选择和注意事项
    因为我本地对FMDB比较熟悉,所以本地数据缓存使用的是FMDB,支持的数据类型如下:
    整数数据类型:integer、bigint、smallint、tinyint
    浮点数据类型:float、double、real
    字符型数据类型:char(n)、varchar(n)、text
    二进制数据:blob
    日期类型:date、time、datetime、timestamp

注意点:

  • 插入或者更新数据的时候对于基本数据类型不能直接插入, 要吧数据封装成NSNumber才行。
  • 查询数据的时候也一样, 拿到结果集然后调用相应的方法转化为相应的数据类型。
  • sql语句要检查数据项个数和顺序是否一致。
  • 使用模拟器调试的时候,每一次重新运行的时候应用的沙盒路径都会改变,尤其是当覆盖运行的时候,本地数据库维护的关系会异常,这个时候需要删除应用重新运行调试。

你可能感兴趣的:(文件下载总结)