1.全称是AFNetworking
2.虽然运行效率没有ASI高,但是使用比ASI简单
3.是对NSURLConnection和NSURLSession的各自的一层包装
AFN的内部中的RunLoop
1.AFN内部开了一条专门用来访问网络请求的线程
2.在这个开线程的方法中,他把方法和dispatch_once都用static修饰了下
3.以保证这个方法的安全性以及只开辟一块内存空间,而且保证他线程不死
4.在这个方法中他会调用另一个网络请求入口的方法
+ (NSThread *)networkRequestThread{
static NSThread *_networkRequestThread = nil;
static dicspatch_once_t oncePredicate;
dispatch_once(&oncePredicate,^{
//创建一个RunLoop
_networkRequestThread = [[NSThread alloc] initWithTarget:self selector:@selector(networkRequestThreadEntryPoint:) object:nil];
[_networkRequestThread start];
});
return __networkRequestThread
}
5.在这个入口方法中他会创建一个RunLoop
6.然后添加一个NSMachPort端口,目的是为了让他里面有Source(因为有了Source的RunLoop才能真正跑起来)
7.然后启动RunLoop,通过RunLoop在里面不断的循环,不断的发送消息,让他做事情.
+ (void)networkRequestThreadEntryPoint:(id)__unused object{
@autoreleasepool{
[[NSThread currentThread] setName:@"AFNetworking"];
NSRunLoop *runLoop = [NSRunLoop currentRunLoop];
[runLoop addPort:[NSMachPort port] forMode:NSDefaultRunLoopMode];
[runLoop run];
}
}
基于NSURLConnection包装的重要对象(iOS9-NSURLConnection已经不能使用)
AFHTTPRequestOperationManager
1.封装了HTTP请求的常用处理
2.GET\POST请求
3.解析服务器的响应数据
//创建
//AFNHTTPRequestOperationManager内部包装了NSURLConnection
AFNHTTPRequestOperationManager *manager = [AFNHTTPRequestOperationManager manager];
//manager内部其实就是alloc init
+ (instancetype)manager{
return [[self alloc] initWithBaseURL:nil];
}
- (instancetype)init{
return [self initWithBaseURL:nil];
}
AFHTTPRequestOperationManager的GET请求
- (void)get{
AFHTTPRequestOperationManager *manager = [AFHTTPRequestOperationManager manager];
NSDictionary *params = @{@"username":@"520",
@"pwd":@"520"
};
[manager GET:@"http://120.25.226.186:32812/login" parameters:params
success:^(AFHTTPRequestOperation *operation, id responseObject){
NSLog(@"成功");
}failure:^(AFHTTPRequestOperation *operation, NSError *error){
NSLog(@"失败");
}];
}
1.参数是不用拼接的,AFN内部会帮你遍历字典然后帮你拼接完成,只需把参数传入即可.
请求成功会来到第一个block,id responseObject这个参数,会自动帮你从服务器得到的JSON数据转为字典或者数组(用id就是因为不一定返回的是什么数据.所以要用到id);他的内部无非也就是一个Request对象
2.NSError *error返回的是错误信息
*AFHTTPRequestOperationManager的POST请求*
- (void)post{
AFHTTPRequestOperationManager *manager = [AFHTTPRequestOperationManager manager];
NSDictionary *params = @{@"username":@"520",
@"pwd":@"520"
};
[manager POST:@"http://120.25.226.186:32812/login" parameters:params
success:^(AFHTTPRequestOperation *operation, id responseObject){
NSLog(@"成功");
}failure:^(AFHTTPRequestOperation *operation, NSError *error){
NSLog(@"失败");
}];
}
1.POST请求和GET没什么两样,无非就是把GET改为POST即可
基于NSURLSession包装的重要对象
AFHTTPSessionManager
1.封装了HTTP请求的常见处理
2.GET\POST请求
3.解析服务器的响应数据
//创建
//AFNHTTPRequestOperationManager内部包装了NSURLConnection
AFNHTTPRequestOperationManager *manager = [AFNHTTPRequestOperationManager manager];
//manager内部其实就是alloc init
+ (instancetype)manager{
return [[self alloc] initWithBaseURL:nil];
}
- (instancetype)init{
return [self initWithBaseURL:nil];
}
AFHTTPSessionManager的GET请求
AFHTTPSessionManager *manager = [AFHTTPSessionManager manager];
NSDictionary *params = @{@"username":@"520",
@"pwd":@"520"
};
[manager GET:@"http://120.25.226.186:32812/login" parameters:params
success:^(NSURLSessionDataTask *task, id responseObject){
NSLog(@"成功");
}failure:^(NSURLSessionDataTask *task, NSError *error){
NSLog(@"失败");
}];
AFHTTPSessionManager的POST请求
AFHTTPSessionManager *manager = [AFHTTPSessionManager manager];
NSDictionary *params = @{@"username":@"520",
@"pwd":@"520"
};
[manager POST:@"http://120.25.226.186:32812/login" parameters:params
success:^(NSURLSessionDataTask *task, id responseObject){
NSLog(@"成功");
}failure:^(NSURLSessionDataTask *task, NSError *error){
NSLog(@"失败");
}];
AFN解析相关
1.AFN在解析的时候:默认解析的是JSON数据。
2.我们通过他的一个内部方法可以看到
AFHTTPSessionManager *manager = [AFHTTPSessionManager manager];
//对相应进行序列化(处理)
manager.responseSerializer
那么有时候服务器返回的是XML数据该怎么办呢
AFHTTPSessionManager *manager = [AFHTTPSessionManager manager];
//对相应进行序列化(处理)
manager.responseSerializer = [AFXMLParserResponseSerializer serializer];
1.也就意味着这里的id responseObject,这里需要换做NSXMLParser *parser
AFHTTPSessionManager *manager = [AFHTTPSessionManager manager];
mgr.responseSerializer = [AFXMLParserResponseSerializer serializer];
NSDictionary *params = @{@"username":@"520",
@"pwd":@"520",
@"type":@"XML"
};
[manager GET:@"http://120.25.226.186:32812/login" parameters:params
success:^(NSURLSessionDataTask *task, NSXMLParser *parser){
//AFN会把解析器提供给你,想要的文件AFN是不知道怎么解析,需要自己来操作
parser.delegate = self;
[parser parse];
}failure:^(NSURLSessionDataTask *task, NSError *error){
NSLog(@"失败");
}];
那么有时候服务器返回的不是JSON也不是XML怎么办呢
AFHTTPSessionManager *manager = [AFHTTPSessionManager manager];
//最普通的解析器
manager.responseSerializer = [AFHTTPResponseSerializer serializer];
[manager GET:@"http://120.25.226.186:32812/login" parameters:params
success:^(NSURLSessionDataTask *task, id responseObject){
NSLog(@"成功");
}failure:^(NSURLSessionDataTask *task, NSError *error){
NSLog(@"失败");
}];
1.这时,就要告诉AFN用最平常的数据来解析,服务器返回的是什么样,就解析成什么样即可
服务器返回的数据解析的几种方式
//解析服务器返回的普通数据(直接使用 *服务器本来返回的数据* 不作任何解析)
mgr.responseSerializer = [AFHTTPResponseSerializer serializer];
//解析服务器返回的XML数据
mgr.responseSerializer = [AFXMLParserResponseSerializer serializer];
//解析服务器返回的JSON数据 (默认解析的是JSON可以不传)
mgr.responseSerializer = [AFJSONRequestSerializer serializer];
项目中的细节处理
有时候,可能当用户在点击一个控制器的button在请求数据时候,会遇到网速慢,返回的数据特别慢,用户可能会等的不耐烦,然后退掉当前的控制器,但是这个请求是还存在着的,一旦请求返回,项目一定会崩溃.那么这种情况该怎么处理呢?
1.遇到这样的情况,可以先把AFNHTTPSessionManger做一个属性
@property(nonatomic, strong)AFNHTTPSessionManger *manger
2.将所有的发请求的地方都用成
// 如果这样使用AFN,会把所有的请求都放入`manger.operationQueue`
self.manger POST/ GET
3.在- (void)dealloc方法中调用
// 取消所有任务
self.manger.operationQueue cancelAllOperations