之前做了文件下载相关的模块,一直忙其他事情,现在终于有时间回顾一下了。文件下载其实分两大块:一是文件本身的下载,二是本地数据关系的维护。
先说文件下载
首先文件下载的需求是前台完整下载、断点下载、后台下载和杀死APP后下载任务保留。需求明确之后回顾一下网络知识。
- NSURLSession: iOS7以后取代之前的NSURLConnection。有两种获取方式。 一种是获取系统提供的单例,这种方式可以共用NSURLCache, NSHTTPCookieStorage 和 NSURLCredentialStorage。第二种方式是自己创建,自己创建的话
+ (NSURLSession *)sessionWithConfiguration:(NSURLSessionConfiguration *)configuration;
+ (NSURLSession *)sessionWithConfiguration:(NSURLSessionConfiguration *)configuration delegate:(nullable id <NSURLSessionDelegate>)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 <NSURLSessionTaskDelegate>: 数据task的代理协议,该协议遵守task代理协议, 并且提供额外的代理方法,包含收到响应、dataTask转变为其他task,收到数据等方法
NSURLSessionDownloadDelegate <NSURLSessionTaskDelegate>: 下载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语句要检查数据项个数和顺序是否一致。
- 使用模拟器调试的时候,每一次重新运行的时候应用的沙盒路径都会改变,尤其是当覆盖运行的时候,本地数据库维护的关系会异常,这个时候需要删除应用重新运行调试。