NSURLSession是iOS7推出的新一代网络框架,在NSURLConnection的基础上多了一种选择,AFNetwrking基于NSURLSession也实现了相应的网络请求API。从功能模块划分,网络可用性,数据的序列化和反序列化以及安全策略这几部分都是一样的,主要是网络请求的管理,请求和返回的处理比前者更加简单,应该说NSURLSession从系统层面实现了NSURLConnection没有实现的部分,比如管理多个请求,直接面向应用层提供上传下载的接口,所以AFNetworking基于NSRULSession实现的网络层更简单,但是功能上没有NSURLConnection那么丰富,比如监听所有任务中完成任务的进度,还有和UIKit的集成。
AFURLSessionManager
NSURLSession
支持多个用task网络请求的管理,是网络层的核心。有NSURLSessionDelegate,NSURLSessionTas
kDelegate,NSURLSessionDataDelegate,NSURLSessionDownloadDelegate 多个代理接口。operationQueue
用于初始化NSURLSession的队列,网络请求返回的回调会在该队列中的线程执行。类比使用NSURLConnection的时候,也有一个OperationQueue,功能基本一样,虽然NSURLConnection自己就是异步线程去发起网络请求,但是如果不使用NSOpeartion和OperationQueue,网络回调都将正在主线程中执行,会影响UI操作的流畅性。在NSURLConnection的实现部分,是通过全局的网络线程去执行回调的。这是NSURLConnection和NSURLSession不太一致的地方,后者是系统层就处理了,比NSURLConnection方便不少。responseSerializer 用于返回数据的反序列化的对象
tasks NSURLSession管理的所有网络请求包含重复的,分为一般的api请求,上传请求和下载请求。
dataTasks 一般的api请求
uploadTasks 上传请求
downloadTasks 下载请求
completionQueue 用户端的回调所在的队列
completionGroup 用户端回调队列所在的组
AFURLSessionManagerTaskDelegagate
在AFURLSessionManager内部用于存储task相关数据的业务对象,详细内容下面会单独介绍。-
mutableTaskDelegatesKeyedByTaskIdentifier 可变字典,用于存储task和对应delegate之间的关联。
剩下的就是在网络请求过程中各种状态的回调block对象了。
AFURLSessionManagerTaskDelegate
在网络请求的各种回调过程中,它用来存储和记录与单个task相关的信息,在网络请求完成之后,将数据
返回给用户端。
AFURLSessionManager
在AFURLSessionManagerTaskDelegate对象中记录的session对象,一个session对象对应这多个
task对象,每个task都有一个与之对应的AFURLSessionManagerTaskDelegate对象。每一个task
在session层面都共享了一些数据,比如序列化对象,回调的队列和组,当一个task完成的时候,都需要
从共享的session中获取这些数据来完成业务层的回调逻辑。mutableData
存储了当前请求获得的返回的数据信息,最后会反序列化成对象或者是以文件的形式保存下来。progress
记录了下载或者上传文件的进度,可以随时通过uploadProgressForTask来获取某个task完成情况,
该方法参数为当前task。AFURLSessionTaskCompletionHandler 由用户端设置的,task请求完成之后的用户端回调block
AFURLSessionDownloadTaskDidFinishDownloadingBlock
由用户端设置的,task下载文件完成之后生成目标文件的保存路径的blockdownloadFileURL
由上一个block生成的目标文件保存路径,在下载完成的回调中设置,然后在整个请求完成的回调中以通知的形式返回给用户端
AFURLSessionManager与AFURLSessionManagerTaskDelegate的关系
在AFURLSessionManager中实现了大部分NSURLSession的协议,而AFURLSessionManagerTaskDelegate也实现了几个重要的协议,比如数据获取,整个task请求完成,文件下载和上传的进度。不同之处在于AFURLSessionManager是针对所有请求的,而taskDelegate只是针对某次具体的请求的。taskDelegate并没有直接从系统层获得回调,而是所有的回调都会先经过AFURLSessionManager处理,然后再分发给每个具体的taskDelegate,由每个taskDelegate自己存储并记录与当前请求相关的数据和信息。至于AFURLSessionManagerTaskDelegate中会有类似以下的回调的实现逻辑:
- (void)URLSession:(__unused NSURLSession *)session
task:(__unused NSURLSessionTask *)task
didSendBodyData:(__unused int64_t)bytesSent
totalBytesSent:(int64_t)totalBytesSent
totalBytesExpectedToSend:(int64_t)totalBytesExpectedToSend
{
self.progress.totalUnitCount = totalBytesExpectedToSend;
self.progress.completedUnitCount = totalBytesSent;
}
task的delegate在这里只获取有用的数据,保存了上传的进度,这样做的目的就是为了保持代码一致性阅读起来更容理解,相当于代理方法的分发。sessionManager里通过字典存下了所有的task,而又在具体的session回调中去处理每个task相关的逻辑。
AFHTTPSessionManager
它继承自AFURLSessionManager,主要功能增加了一些初始化的方法以及对序列化和反序列化阶段的错误检查,还有设置具体的序列化对象。只是在AFURLSessionManager的基础上做了更容易使用的接口的封装。
NSURLSession和NSURLConnection
AFNetworking中NSRULSession的实现上的设计上基本是一致的,AFURLSessionManager相当于AFHTTPRequestOperationManager。而AFURLConnectionOperation类似与NSURLSessionTask。但是对网络回调的实现上是不一致的,前者是依赖NSURLSession处理,然后分发给每个task自己处理。而NSURLConnection是每个请求都是独立处理各自来自系统层的网络回调,这是由于采用了不同的系统网络框架造成的。两种实现都有一个问题,就是如果baseURL是经常变化的,下载某一个素材的时候,这个时候比较好的办法是只使用NSRULConnection中的AFURLConnectionPeration,每次url由自己指定,自己去维护OperationQueue,可能需要自己实现AFHTTPRequestOperationManager的部分。