AFN 的基础部分是 AFURLConnectionOperation,一个 NSOperation 子类,实现了 基于NSURLConnection 相关的delegate+blocks,网络部分是由 NSURLConnection 完成,然后利用 NSOperation 的 state(isReady→isExecuting→isFinished) 变化来进行网络控制。网络请求是在一个指定的线程(networkRequestThread)完成。
AFURLConnectionOperation是一个很纯粹的网络请求 operation,可以对他进行 start/cancel/pause/resume 操作,可以获取对应的 NSURLRequest 和 NSURLResponse 数据。提供了 uploadPress 和downloadProgress 以方便其他使用。
AFHTTPRequestOperation是 AFURLConnectionOperation 的子类,针对 HTTP+HTTPS 协议做了一层封装,比如statusCode、Content-Type 等,添加了请求成功和失败的回调 block,提供了addAcceptableContentTypes: 以方便上层使用。
NSURLSession
NSURLSession 是 iOS 7 新引入的用于替代 NSURLConnection 的类。NSURLConnection 并没有被弃用,今后一段时间应该也不会(事实上最终还是被弃用了)。它们有一些重叠,AFNetworking提供了更高层次的抽象,并最大程度扩展了它的实用性。
模块化
对于AFNetworking的主要批评之一是笨重。虽然它的构架使在类的层面上是模块化的,但它的包装并不允许选择独立的一些功能。随着时间的推移,AFHTTPClient 尤其变得不堪重负。 在AFNetworking 2.0 中,你可以挑选并通过 CocoaPods subspecs 选择你所需要的组件
AFN 3.0+弃用NSURLConnection,只提供NSURLSession的支持。
NSData *imageData = UIImagePNGRepresentation(image);
//1.NSURL
NSURL *url = [NSURL URLWithString:@"http://localhost/post/upload.php"];
//2.NSMutableURLRequest
NSMutableURLRequest *request = [NSMutableURLRequest requestWithURL:url];
//2.1 改请求方法
request.HTTPMethod = @"POST";
//2.2.设置请求头,boundary 这个就是一个分隔符,可以随便写, 只要不是中文
NSString *contentTypevalue = [NSString stringWithFormat:@"multipart/form-data; boundary=%@", kBoundaryUpload];
[request setValue:contentTypevalue forHTTPHeaderField:@"Content-Type"];
//NSURLSession不要把上传的文件放入请求体中,应该放在NSURLSession的方法中
NSData *fileData = [self formDataWithFileData:imageData serverFieldName:@"userfile" fileSaveName:@"abc.png"];
//3.获取系统提供的NSURLSession
NSURLSession *globalSession = [NSURLSession sharedSession];
//4.由session发起上传任务
[[globalSession uploadTaskWithRequest:request fromData:fileData completionHandler:^(NSData * _Nullable data, NSURLResponse * _Nullable response, NSError * _Nullable error) {
//4.反序列化
if (error==nil && data.length>0) {
id result =[NSJSONSerialization JSONObjectWithData:data options:0 error:NULL];
NSLog(@"login-----%@",result);
}
}] resume];
- (NSData *)formDataWithFileData:(NSData *)fileData serverFieldName:(NSString *)serverFieldName fileSaveName:(NSString *)fileSaveName{
//1.NSMutableData
NSMutableData *dataM = [NSMutableData data];
/**
-----------------------------11454065798049615451989194362
Content-Disposition: form-data; name="userfile"; filename="abc.txt"
Content-Type: application/octet-stream
111222333
-----------------------------11454065798049615451989194362--
*/
//2.拼接头
NSMutableString *headerString = [NSMutableString string];
//2.1 拼接第一行
[headerString appendFormat:@"--%@\r\n",kBoundaryUpload];
//2.1 拼接第二行
[headerString appendFormat:@"Content-Disposition: form-data; name=%@; filename=%@\r\n",serverFieldName,fileSaveName];
//2.3 拼接第三行
[headerString appendString:@"Content-Type: application/octet-stream\r\n\r\n"];
//2.4 将头部的字符串转成二进制
NSData *headerData = [headerString dataUsingEncoding:NSUTF8StringEncoding];
//2.5 拼接到dataM后面
[dataM appendData:headerData];
//3.拼接内容
[dataM appendData:fileData];
[dataM appendData:[@"\r\n" dataUsingEncoding:NSUTF8StringEncoding]];
//4.拼接尾巴
NSString *tailString = [NSString stringWithFormat:@"--%@--\r\n\r\n",kBoundaryUpload];
[dataM appendData:[tailString dataUsingEncoding:NSUTF8StringEncoding]];
return dataM.copy;
}
- (NSURLSession *)downLoadSession{
if (_downLoadSession==nil) {
//开发中,就用这个
NSURLSessionConfiguration *defaultConfigration = [NSURLSessionConfiguration defaultSessionConfiguration];
/**
参数1:配置
参数2:代理
参数3:队列,如果我们写的是主队列,那么代理方法在主线程调用
如果是通过 [[NSOperationQueue] alloc] init]那么代理方法在子线程调用
nil 这相当于 [[NSOperationQueue] alloc] init]
*/
_downLoadSession = [NSURLSession sessionWithConfiguration:defaultConfigration delegate:self delegateQueue:nil];
}
return _downLoadSession;
}
- (void)touchesBegan:(NSSet *)touches withEvent:(UIEvent *)event{
//1.NSURL
NSURL *url = [NSURL URLWithString:@"http://localhost/videos.zip"];
//2.下载我们为了设置代理,获取进度,自己创建session
NSURLSession *downLoadSession = self.downLoadSession;
//3.由downLoadSession发起下载任务
NSURLSessionDownloadTask *downLoadTask = [downLoadSession downloadTaskWithURL:url];
//4.resume
[downLoadTask resume];
}
AFN内部开了一条专门用来访问网络请求的线程,在这个开线程的方法中,他把方法和dispatch_once都用static修饰了下,以保证这个方法的安全性以及只开辟一块内存空间,而且保证他线程不死,在这个方法中他会调用另一个网络请求入口的方法
在这个入口方法中他会创建一个RunLoop,然后添加一个NSMachPort端口,目的是为了让他里面有Source(因为有了Source的RunLoop才能真正跑起来)
然后启动RunLoop,通过RunLoop在里面不断的循环,不断的发送消息,让他做事情.
AFHTTPSessionManager封装了HTTP请求的常见处理,包括GET/POST方法,同时还包括了解析服务器响应数据的方法
AFHTTPSessionManager的GET请求
- (NSURLSessionDataTask *)GET:(NSString *)URLString
parameters:(id)parameters
success:(void (^)(NSURLSessionDataTask *task, id responseObject))success
failure:(void (^)(NSURLSessionDataTask *task, NSError *error))failure
AFHTTPSessionManager的POST请求
- (NSURLSessionDataTask *)POST:(NSString *)URLString
parameters:(id)parameters
success:(void (^)(NSURLSessionDataTask *task, id responseObject))success
failure:(void (^)(NSURLSessionDataTask *task, NSError *error))failure
AFN解析相关
AFN在解析时候,默认解析的是JSON数据。如果想解析XML数据,就需要手动把responseSerializer
的值该掉
AFHTTPSessionManager *mgr = [AFHTTPSessionManager manager];
mgr.responseSerializer = [AFXMLParserResponseSerializer serializer];
将方法的返回值类型也改成XML,然后自己解析。
如果服务器返回的是普通的数据类型,这时就要告诉AFN用普通的数据类型来解析,服务器返回什么样,解析成什么样即可
AFHTTPSessionManager *mgr = [AFHTTPSessionManager manager];
mgr.responseSerializer = [AFHTTPResponseSerializer serializer];