AFNetWoking(3.0)是一个著名的iOS第三方网络通信框架,当前最新的3.0版的实现完全基于NSURLSession进行了封装,作为一个刚刚学习NSURLSession的小白,本文主要关注AFNetWorking(3.0)中使用NSURLSession进行网络通信的逻辑。
NSURLSession是苹果公司在iOS7.0后推出用来替代NSURLConnection进行网络请求的API,有关NSURLSession的使用在我对官方文档翻译后总结的NSURLSession的使用中有介绍和说明。
AFNetWorking(3.0)中进行网络通信的类
在AFNetWorking(3.0)中,实现网络通信功能的类主要有三个
- AFHTTPSessionManager
- AFURLSessionManager
- AFURLSessionManagerTaskDelegate
其中AFHTTPSessionManager是AFURLSessionManager的子类,它封装了用户对于HTTP请求参数的操作,用户通过调用相关方法就能进行GET、POST、PUT等方法;
AFURLSessionManager是AFNetWorking(3.0)的核心,它封装了几乎所有与NSURLSession相关的方法,并实现了NSURLSessionDelegate、 NSURLSessionTaskDelegate,、NSURLSessionDataDelegate、NSURLSessionDownloadDelegate等委托的方法。
AFURLSessionManagerTaskDelegate也实现委托的方法,不过仅仅实现了部分,它的作用主要是管理上传与下载任务的进度。
AFNetWorking(3.0)的网络通信流程
1. 创建一个AFURLSessionManager对象
AFURLSessionManager封装了关于NSURLSession对象的操作,因此通过创建一个AFURLSessionManager对象就简化了创建NSURLSession的流程。
- (instancetype)initWithSessionConfiguration:(NSURLSessionConfiguration *)configuration {
self = [super init];
if (!self) {
return nil;
}
//1、config为空设为默认config
if (!configuration) {
configuration = [NSURLSessionConfiguration defaultSessionConfiguration];
}
self.sessionConfiguration = configuration;
//2、创建操作队列,设置最大并发数
self.operationQueue = [[NSOperationQueue alloc] init];
self.operationQueue.maxConcurrentOperationCount = 1;
//3、创建session
self.session = [NSURLSession sessionWithConfiguration:self.sessionConfiguration delegate:self delegateQueue:self.operationQueue];
//4、创建响应序列化,安全策略,任务识别关键字
self.responseSerializer = [AFJSONResponseSerializer serializer];
self.securityPolicy = [AFSecurityPolicy defaultPolicy];
#if !TARGET_OS_WATCH
self.reachabilityManager = [AFNetworkReachabilityManager sharedManager];
#endif
self.mutableTaskDelegatesKeyedByTaskIdentifier = [[NSMutableDictionary alloc] init];
//5、创建锁
self.lock = [[NSLock alloc] init];
self.lock.name = AFURLSessionManagerLockName;
//6、为session管理的所有任务设置关联的AFURLSessionManagerTaskDelegate对象
[self.session getTasksWithCompletionHandler:^(NSArray *dataTasks, NSArray *uploadTasks, NSArray *downloadTasks) {
//1、数据任务
for (NSURLSessionDataTask *task in dataTasks) {
[self addDelegateForDataTask:task uploadProgress:nil downloadProgress:nil completionHandler:nil];
}
//2、上传任务
for (NSURLSessionUploadTask *uploadTask in uploadTasks) {
[self addDelegateForUploadTask:uploadTask progress:nil completionHandler:nil];
}
//3、下载任务
for (NSURLSessionDownloadTask *downloadTask in downloadTasks) {
[self addDelegateForDownloadTask:downloadTask progress:nil destination:nil completionHandler:nil];
}
}];
return self;
}
通过上述代码可以看到AFURLSessionManager的初始化做了以下工作:
- 初始化会话配置,默认的会话配置为defaultSessionConfiguration类型
- 初始化操作队列,队列并发数为1,即为串行队列
- 通过会话配置和操作队列创建NSURLSession对象,并将自身设为委托
- 创建网络响应序列化工具对象
- 定义了网络通信的安全策略
- 初始化映射数组,在进行多个网络请求任务时,每个task都对应着一个AFURLSessionManagerTaskDelegate对象
- 为session管理的所有任务设置关联的AFURLSessionManagerTaskDelegate对象
2. 封装网络通信请求参数
有了NSURLSession对象之后,便可以进行网络通信了,iOS提供了两种方式,一种是通过URL直接进行网络请求,一种是采用NSURLRequest对象进行请求;AFNetWorking将请求统一封装成NSURLRequest对象。
使用NSURLSession进行网络请求,需要选择相应的任务类型:数据任务类型(NSURLSessionDataTask)、上传任务类型(NSURLSessionUploadTask)和下载任务类型(NSURLSessionDownloadTask)。
对于数据任务类型,AFNetWorking(3.0)通过建立子类AFHTTPSessionManager对AFURLSessionManager类进行了进一步的抽象和封装。这样用户在使用时,针对请求方法的不同选择不同的网络请求方法,只需传入以下参数便可进行网络请求:
- 请求地址
- 请求参数
- 处理请求成功的block
- 处理请求失败的block
- 处理上传进度block
- 处理下载进度block
虽然AFHTTPSessionMananger继承于AFURLSessionManager,但由于有关NSURLSession的配置与操作全部封装在AFURLSessionManager中,所以AFHTTPSessionMananger所要做的工作其实并不多,主要是
- 根据传入的URL和请求参数创建和修改NSURLResquest对象
- 根据用户选择的网络请求方法,调用NSURLSessionManager对应的方法,并对返回的NSURLSessionDataTask进行操作,所以AFHTTPSessionManager主要是用来进行数据任务的,有关上传和下载的任务并不AFHTTPSessionManager中进行。
在AFHTTPSessionManager中仅包含三个属性:
- NSString类型的URL
- 对请求进行序列化操作的AFHTTPRequestSerializer对象,封装了对NSURLResquest对象进行操作的方法
- 对响应进行序列化操作的AFHTTPResponseSerializer对象,封装了对NSURLResponse对象进行操作的方法
@property (readonly, nonatomic, strong, nullable) NSURL *baseURL;
@property (nonatomic, strong) AFHTTPRequestSerializer <AFURLRequestSerialization> * requestSerializer;
@property (nonatomic, strong) AFHTTPResponseSerializer <AFURLResponseSerialization> * responseSerializer;
AFHTTPSessionMananger提供了一些初始化的方法,这些方法都是基于如下方法:
- (instancetype)initWithBaseURL:(NSURL *)url sessionConfiguration:(NSURLSessionConfiguration *)configuration
该方法对AFHTTPSessionMananger进行初始化操作,并传入参数configuration给AFURLSessionManager的初始化方法
AFHTTPSessionMananger提供了常用的网络请求方法,如:GET、POST、HEAD、PUT、PATCH、DELETE方法,用户所能调用方法都基于AFHTTPSessionMananger的以下两个方法:
//进行网络请求操作
- (NSURLSessionDataTask *)dataTaskWithHTTPMethod:(NSString *)method
URLString:(NSString *)URLString
parameters:(id)parameters
uploadProgress:(nullable void (^)(NSProgress *uploadProgress)) uploadProgress
downloadProgress:(nullable void (^)(NSProgress *downloadProgress)) downloadProgress
success:(void (^)(NSURLSessionDataTask *, id))success
failure:(void (^)(NSURLSessionDataTask *, NSError *))failure;
//进行请求体包含block的POST请求
- (NSURLSessionDataTask *)POST:(NSString *)URLString
parameters:(id)parameters
constructingBodyWithBlock:(void (^)(id <AFMultipartFormData> formData))block
progress:(nullable void (^)(NSProgress * _Nonnull))uploadProgress
success:(void (^)(NSURLSessionDataTask *task, id responseObject))success
failure:(void (^)(NSURLSessionDataTask *task, NSError *error))failure;
对于上传和下载任务,开发者可以采用同样的方式对请求参数进行封装,或者直接调用AFURLSessionManager中的请求方法。
//上传文件请求
- (NSURLSessionUploadTask *)uploadTaskWithRequest:(NSURLRequest *)request
fromFile:(NSURL *)fileURL
progress:(void (^)(NSProgress *uploadProgress)) uploadProgressBlock
completionHandler:(void (^)(NSURLResponse *response, id responseObject, NSError *error))completionHandler;
//上传数据类型
- (NSURLSessionUploadTask *)uploadTaskWithRequest:(NSURLRequest *)request
fromData:(NSData *)bodyData
progress:(void (^)(NSProgress *uploadProgress)) uploadProgressBlock
completionHandler:(void (^)(NSURLResponse *response, id responseObject, NSError *error))completionHandler;
//上传数据流
- (NSURLSessionUploadTask *)uploadTaskWithStreamedRequest:(NSURLRequest *)request
progress:(void (^)(NSProgress *uploadProgress)) uploadProgressBlock
completionHandler:(void (^)(NSURLResponse *response, id responseObject, NSError *error))completionHandler;
//下载请求
- (NSURLSessionDownloadTask *)downloadTaskWithRequest:(NSURLRequest *)request
progress:(void (^)(NSProgress *downloadProgress)) downloadProgressBlock
destination:(NSURL * (^)(NSURL *targetPath, NSURLResponse *response))destination
completionHandler:(void (^)(NSURLResponse *response, NSURL *filePath, NSError *error))completionHandler;
//恢复下载请求
- (NSURLSessionDownloadTask *)downloadTaskWithResumeData:(NSData *)resumeData
progress:(void (^)(NSProgress *downloadProgress)) downloadProgressBlock
destination:(NSURL * (^)(NSURL *targetPath, NSURLResponse *response))destination
completionHandler:(void (^)(NSURLResponse *response, NSURL *filePath, NSError *error))completionHandler;
3. NSURLSessionTask的创建
要使用NSURLSession对象进行网络通信,需要用该对象调用请求方法返回一个NSURLSessionTask对象,通过task对象开启网络请求。AFURLSessionManager会在task开启网络请求之前,将task与AFURLSessionManagerTaskDelegate对象通过字典mutableTaskDelegatesKeyedByTaskIdentifier进行映射,以确保每一个task对象都会有一个taskDelegate对象。
AFURLSessionManagerTaskDelegate类的定义在AFURLSessionManager中,它的作用主要是管理网络请求任务的进度,因此也实现了NSURLSessionTaskDelegate、 NSURLSessionDataDelegate、NSURLSessionDownloadDelegate的部分方法。task对象与taskDelegate对象进行映射的过程如下:
- 调用NSURLSession对象的方法返回一个task对象
- 创建一个taskDelegate对象,将传入的有关进度和出路请求完成的block赋给taskDelegate对象
- 以task对象的taskIdentifier标识符为key,将taskDelegate对象添加到字典中
这里有一个问题需要解决,就是如何保证task对象的标识符是唯一的?
在第一节中初始化AFURLSessionManager对象时,创建session对象时的委托是self,也就是AFURLSessionManager对象本身,AFNetWorking(3.0)在进行网络请求时,所有的task对象都由该session对象产生,而由同一个session产生的多个task,它们的标识符都是唯一的;不同的session产生的task,task的标识符则不一定唯一。这样做就能够保证在进行多个网络请求时,对每个网络请求的task进行单独管理,例如多个下载任务在执行,用户可以决定哪个任务暂停,哪个任务继续下载。字典添加映射代码如下,以下载为例:
- (void)addDelegateForDownloadTask:(NSURLSessionDownloadTask *)downloadTask
progress:(void (^)(NSProgress *downloadProgress)) downloadProgressBlock
destination:(NSURL * (^)(NSURL *targetPath, NSURLResponse *response))destination
completionHandler:(void (^)(NSURLResponse *response, NSURL *filePath, NSError *error))completionHandler
{
//1、创建一个AFURLSessionManagerTaskDelegate对象,将block参数赋给delegate
AFURLSessionManagerTaskDelegate *delegate = [[AFURLSessionManagerTaskDelegate alloc] init];
delegate.manager = self;
delegate.completionHandler = completionHandler;
//2、返回task对象
if (destination) {
delegate.downloadTaskDidFinishDownloading = ^NSURL * (NSURLSession * __unused session, NSURLSessionDownloadTask *task, NSURL *location) {
return destination(location, task.response);
};
}
downloadTask.taskDescription = self.taskDescriptionForSessionTasks;
//3、将task对象与delegate进行关联
[self setDelegate:delegate forTask:downloadTask];
delegate.downloadProgressBlock = downloadProgressBlock;
}
- (void)setDelegate:(AFURLSessionManagerTaskDelegate *)delegate
forTask:(NSURLSessionTask *)task
{
NSParameterAssert(task);
NSParameterAssert(delegate);
[self.lock lock];
//以task的标识符为key,将delegate放入字典
self.mutableTaskDelegatesKeyedByTaskIdentifier[@(task.taskIdentifier)] = delegate;
[delegate setupProgressForTask:task];
[self addNotificationObserverForTask:task];
[self.lock unlock];
}
这里又出现了另一个问题,既然所有的task都是由同一个session产生,而该session对象的委托确是AFURLSessionManager对象本身,网络请求的执行会调用AFURLSessionManager实现的方法,如何通过AFURLSessionManagerTaskDelegate对象来控制网络请求任务的进行?
前面说过,AFURLSessionManagerTaskDelegate实现了委托的部分方法,这部分方法控制着网络请求的任务进度,而在AFURLSessionManager中也实现了这些方法,当系统调用AFURLSessionManager中的实现方法时,AFURLSessionManager根据task的标识符在字典中找到对应的AFURLSessionManagerTaskDelegate对象,若AFURLSessionManagerTaskDelegate对象中实现了该方法,则调用AFURLSessionManagerTaskDelegate中实现的方法。
4. 网络请求的进度管理
网络请求的进度管理是由AFURLSessionManagerTaskDelegate负责的,AFURLSessionManagerTaskDelegate具有以下属性:
@property (nonatomic, weak) AFURLSessionManager *manager; //指向AFURLSessionManager对象,能够调用AFURLSessionManager中的block方法
@property (nonatomic, strong) NSMutableData *mutableData; //存放网络请求接收到的数据
@property (nonatomic, strong) NSProgress *uploadProgress; //上传任务进度
@property (nonatomic, strong) NSProgress *downloadProgress; //下载任务进度
@property (nonatomic, copy) AFURLSessionDownloadTaskDidFinishDownloadingBlock downloadTaskDidFinishDownloading; //下载完成回调
@property (nonatomic, copy) AFURLSessionTaskProgressBlock uploadProgressBlock; //上传任务进度回调
@property (nonatomic, copy) AFURLSessionTaskProgressBlock downloadProgressBlock; //下载任务进度回调
@property (nonatomic, copy) AFURLSessionTaskCompletionHandler completionHandler; //处理回调
当网络进行网络通信时,AFURLSessionManagerTaskDelegate中实现的委托方法会返回进度信息并记录到uploadProgress和downloadProgress中,同时接收到的数据也会存放到mutableData中。为了能够响应进度信息的变化,AFURLSessionManagerTaskDelegate则通过KVO的对进度信息监听,一旦进度信息发生改变,则可以调用回调进行处理,例如可以在回调中刷新主界面。这些回调函数都是在作为网络请求的参数传入到AFURLSessionManager中,并由AFURLSessionManager创建任务时赋给AFURLSessionManagerTaskDelegate对象的。
//设置uploadProgress与downloadProgress,并添加观察者
- (void)setupProgressForTask:(NSURLSessionTask *)task {
//#1、对downloadProgress和uploadProgress进行设置,代码略
//#2、添加对downloadProgress和uploadProgress的KVO观察者
[self.downloadProgress addObserver:self
forKeyPath:NSStringFromSelector(@selector(fractionCompleted))
options:NSKeyValueObservingOptionNew
context:NULL];
[self.uploadProgress addObserver:self
forKeyPath:NSStringFromSelector(@selector(fractionCompleted))
options:NSKeyValueObservingOptionNew
context:NULL];
}
//移除观察者
- (void)cleanUpProgressForTask:(NSURLSessionTask *)task {
[self.downloadProgress removeObserver:self forKeyPath:NSStringFromSelector(@selector(fractionCompleted))];
[self.uploadProgress removeObserver:self forKeyPath:NSStringFromSelector(@selector(fractionCompleted))];
}
//添加默认的监听方法
- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary<NSString *,id> *)change context:(void *)context {
//调用downloadProgress的回调
if ([object isEqual:self.downloadProgress]) {
if (self.downloadProgressBlock) {
self.downloadProgressBlock(object);
}
}
//调用uploadProgress的回调
else if ([object isEqual:self.uploadProgress]) {
if (self.uploadProgressBlock) {
self.uploadProgressBlock(object);
}
}
}
当NSURLSessionTask结束时,总会调用委托方法URLSession:task:didCompleteWithError: ,该方法告诉委托当前的网络数据通信已完成。在AFNetWorking(3.0)中,该方法的主要工作如下:
- 从manager中获取本次请求的response序列化信息,放入userInfo字典
- 将网络通信接收的数据放入userInfo字典
- 若网络通信错误,则将error放入userInfo字典,同时使用处理回调进行后续处理,并发送通知
- 若网络通信正常,则将response序列化信息进行解析,放入userInfo字典中,同时调用处理回调进行后续处理,并发送通知
- (void)URLSession:(__unused NSURLSession *)session
task:(NSURLSessionTask *)task
didCompleteWithError:(NSError *)error
{
__strong AFURLSessionManager *manager = self.manager;
__block id responseObject = nil;
__block NSMutableDictionary *userInfo = [NSMutableDictionary dictionary];
userInfo[AFNetworkingTaskDidCompleteResponseSerializerKey] = manager.responseSerializer;
//Performance Improvement from #2672
NSData *data = nil;
if (self.mutableData) {
data = [self.mutableData copy];
//We no longer need the reference, so nil it out to gain back some memory.
self.mutableData = nil;
}
if (self.downloadFileURL) {
userInfo[AFNetworkingTaskDidCompleteAssetPathKey] = self.downloadFileURL;
} else if (data) {
userInfo[AFNetworkingTaskDidCompleteResponseDataKey] = data;
}
if (error) {
userInfo[AFNetworkingTaskDidCompleteErrorKey] = error;
dispatch_group_async(manager.completionGroup ?: url_session_manager_completion_group(), manager.completionQueue ?: dispatch_get_main_queue(), ^{
if (self.completionHandler) {
self.completionHandler(task.response, responseObject, error);
}
dispatch_async(dispatch_get_main_queue(), ^{
[[NSNotificationCenter defaultCenter] postNotificationName:AFNetworkingTaskDidCompleteNotification object:task userInfo:userInfo];
});
});
} else {
dispatch_async(url_session_manager_processing_queue(), ^{
NSError *serializationError = nil;
responseObject = [manager.responseSerializer responseObjectForResponse:task.response data:data error:&serializationError];
if (self.downloadFileURL) {
responseObject = self.downloadFileURL;
}
if (responseObject) {
userInfo[AFNetworkingTaskDidCompleteSerializedResponseKey] = responseObject;
}
if (serializationError) {
userInfo[AFNetworkingTaskDidCompleteErrorKey] = serializationError;
}
dispatch_group_async(manager.completionGroup ?: url_session_manager_completion_group(), manager.completionQueue ?: dispatch_get_main_queue(), ^{
if (self.completionHandler) {
self.completionHandler(task.response, responseObject, serializationError);
}
dispatch_async(dispatch_get_main_queue(), ^{
[[NSNotificationCenter defaultCenter] postNotificationName:AFNetworkingTaskDidCompleteNotification object:task userInfo:userInfo];
});
});
});
}
}
以上就是我阅读AFNetWorking(3.0)的源码后做的关于NSURLSession的使用的总结。AFNetWorking(3.0)作为一个成熟、稳定的网络请求框架,所涉及的知识也不局限于NSURLSession的使用,还有许多内容,如网络状态的监测、网络请求的安全性处理,这些都非常值得我们花时间去学习。