文章目录

  • 前言
  • 3.0相较于2.0
  • 删除的类
  • 替换的类
  • 参数设置
  • 具体创建流程
  • 进行网络请求
  • GET请求
  • POST请求
  • 文件上传
  • 请求结果的处理
  • 文件下载
  • 监听网络状态
  • 更换头像遇到的问题


前言

这里用到的是AFNetworking 3.0 进行文件上传和文件下载的。

AFNetWorking 3.0是AFNetWorking最新的主要发布版,iOS和Mac OS X中令人感到兴奋的网络库。为了保持版本向前发展的可维护性,3.0中删除了所有基于被废弃的NSURLConnection的API。如果你的项目使用到了基于NSURLConnection的API,建议你使用基于NSURLSession的API。

3.0相较于2.0

  • 在AFNetworking 3.0之前,底层是通过封装NSURLConnection来实现的。
  • 在AFNetworking 3.0之后,也就是在iOS 9.0 之后,NSURLConnection被弃用,苹果推荐使用NSURLSession来管理网络请求,所以AFNetworking 3.0之后,底层是通过封装NSURLSession来实现的。

删除的类

关于以下的类已经被删除:

  • AFURLConnectionOperation
  • AFHTTPRequestOperation
  • AFHTTPRequestOperationManager

因此在其他地方上搜到的使用AFHTTPRequestOperationManager进行请求的方法在AFNetworking3.0中都已经无法使用。

替换的类

上述被删除的类依次被下面三个类代替了,同时请求方法也跟着改变了,所以AFNetworking 3.0以后发生了很大的变化。

  • AFURLSessionManager
  • AFHTTPSessionManager
  • AFNetworkReachabilityManager

有一些类中的关键实现是通过NSURLConnection实现的,他们内部已经使用NSURLSession进行了重构。

参数设置

AFNetworking 3.0 使用的是 AFHTTPSessionManager 来进行网络请求的,下面介绍一下参数的设置。

  • 关于 requestSerializer 它是 AFNetworking 参数编码的序列化器,它一共有三种编码格式,responseSerializer 同理,选择哪种看自己的需求。
  1. AFHTTPRequestSerializer:是普通的 HTTP 的编码格式的,也就是 mid=10&method=userInfo&dateInt=20160818 这种格式的。
  2. AFJSONRequestSerializer:是 JSON 编码格式的,也就是 {"mid":"11","method":"userInfo","dateInt":"20160818"} 这种格式的。
  3. AFPropertyListRequestSerializer:这个没用过,但是看介绍说是编码成 plist 格式的参数。
  • HTTPHeaderField 请求头没有的话可以不用设置。

具体创建流程

  1. 创建网络请求对象
  2. 设置请求和接收的数据编码格式
  3. 设置请求头 (没有可以不需要设置)
  4. 请求地址
  5. 请求参数
/* 创建网络请求对象 */
AFHTTPSessionManager *manager = [AFHTTPSessionManager manager];
  
/* 设置请求和接收的数据编码格式 */
manager.requestSerializer = [AFJSONRequestSerializer serializer]; // 设置请求数据为 JSON 数据
manager.responseSerializer = [AFJSONResponseSerializer serializer]; // 设置接收数据为 JSON 数据
  
/* 设置请求头 */
[manager.requestSerializer setValue:@"application/json" forHTTPHeaderField:@"Content-Type"];
[manager.requestSerializer setValue:@"xxx" forHTTPHeaderField:@"xxx"];
[manager.requestSerializer setValue:@"xxx" forHTTPHeaderField:@"xxx"];
  
/* 请求地址 */
NSString *url = @"http://xxx";
  
/* 请求参数 */
NSDictionary *parameters = @{@"xxx" : @"xxx",
                             @"xxx" : @"xxx",
                             @"xxx" : @"xxx"};

进行网络请求

网络请求有 GET 请求、POST 请求、文件上传 这几种,按自己需求选择相应请求的方法,参数设置使用上面的设置具体流程,文件下载 的话有点不一样,下面会单独说。

GET请求

[manager GET:url parameters:parameters progress:nil success:^(NSURLSessionDataTask * _Nonnull task, id  _Nullable responseObject) {
    
    NSLog(@"请求成功:%@", responseObject);
    
} failure:^(NSURLSessionDataTask * _Nullable task, NSError * _Nonnull error) {
    
    NSLog(@"请求失败:%@", error);
}];

POST请求

[manager POST:url parameters:parameters progress:nil success:^(NSURLSessionDataTask * _Nonnull task, id  _Nullable responseObject) {
    
    NSLog(@"请求成功:%@", responseObject);
    
} failure:^(NSURLSessionDataTask * _Nullable task, NSError * _Nonnull error) {
    
    NSLog(@"请求失败:%@", error);
}];

文件上传

下面以图片上传为例,其他文件的上传都是差不多的。

[manager POST:url parameters:parameters constructingBodyWithBlock:^(id<AFMultipartFormData>  _Nonnull formData) {
    
    /* 本地图片上传 */
    NSURL *imageUrl = [[NSBundle mainBundle] URLForResource:@"test" withExtension:@"png"];
    NSData *imageData = [NSData dataWithContentsOfURL:imageUrl];
    
    // 直接将图片对象转成 data 也可以
    // UIImage *image = [UIImage imageNamed:@"test"];
    // NSData *imageData = UIImageJPEGRepresentation(image, 1.0);
    
    /* 上传数据拼接 */
    [formData appendPartWithFileData:imageData name:@"file" fileName:@"test.png" mimeType:@"image/png"];
    
} progress:nil success:^(NSURLSessionDataTask * _Nonnull task, id  _Nullable responseObject) {
    
    NSLog(@"上传成功:%@", responseObject);
    
} failure:^(NSURLSessionDataTask * _Nullable task, NSError * _Nonnull error) {
    
    NSLog(@"上传失败:%@", error);
}];

请求结果的处理

  • 以下方法能够将获取到的 responseObject 数据直接转成 NSString 字符串或者 NSDictionary 字典,也可以使用 MJExtension 这个第三方库,能够将获取到的 responseObject 直接转换成对象,如果要转成字典或是对象, responseObject 必须是 JSON 格式的。
[manager POST:url parameters:parameters progress:nil success:^(NSURLSessionDataTask * _Nonnull task, id  _Nullable responseObject) {
    
    /* 将得到的数据转换成 NSString 字符串 */
    NSString *infoStr = [[NSString alloc] initWithData:responseObject encoding:NSUTF8StringEncoding];
    NSLog(@"请求到的数据:%@", infoStr);
    
    /* 将得到的 JSON 数据转换成 NSDictionary 字典 */
    NSDictionary *resultDic = [NSJSONSerialization JSONObjectWithData:responseObject options:NSJSONReadingMutableLeaves error:nil];
    NSLog(@"请求到的数据:%@", resultDic);
    
} failure:^(NSURLSessionDataTask * _Nullable task, NSError * _Nonnull error) {
    
    NSLog(@"请求失败:%@", error);
}];
  • AFNetworking 2.0 的返回参数是 AFHTTPRequestOperation ,而 AFNetworking 3.0 的返回参数是 NSURLSessionDataTask ,所以还是有些区别的,如果需要得到 allHeaderFields 这个参数需要强转一下。
[manager POST:url parameters:parameters progress:nil success:^(NSURLSessionDataTask * _Nonnull task, id  _Nullable responseObject) {
    
    /* allHeaderFields 参数获取 */
    NSHTTPURLResponse *response = (NSHTTPURLResponse *)task.response;
    NSDictionary *headerFields = response.allHeaderFields;
    NSLog(@"请求头:%@", headerFields);
    
} failure:^(NSURLSessionDataTask * _Nullable task, NSError * _Nonnull error) {
    
    NSLog(@"请求失败:%@", error);
}];

文件下载

文件下载的话和上面POST、GET、文件上传三个不太一样,所以单独来说,上面那些设置的参数都不需要了,具体代码看下面。

/* 创建网络下载对象 */
AFURLSessionManager *manager = [[AFURLSessionManager alloc] initWithSessionConfiguration:[NSURLSessionConfiguration defaultSessionConfiguration]];
  
/* 下载地址 */
NSURL *url = [NSURL URLWithString:@"http://xxx/test.mp4"];
NSURLRequest *request = [NSURLRequest requestWithURL:url];
  
/* 下载路径 */
NSString *path = [NSHomeDirectory() stringByAppendingPathComponent:@"Documents"];
NSString *filePath = [path stringByAppendingPathComponent:url.lastPathComponent];
  
/* 开始请求下载 */
NSURLSessionDownloadTask *downloadTask = [manager downloadTaskWithRequest:request progress:^(NSProgress * _Nonnull downloadProgress) {
    
    NSLog(@"下载进度:%.0f%", downloadProgress.fractionCompleted * 100);
    
} destination:^NSURL * _Nonnull(NSURL * _Nonnull targetPath, NSURLResponse * _Nonnull response) {
    
    /* 设定下载到的位置 */
    return [NSURL fileURLWithPath:filePath];
    
} completionHandler:^(NSURLResponse * _Nonnull response, NSURL * _Nullable filePath, NSError * _Nullable error) {
    
    NSLog(@"下载完成");
    
}];
[downloadTask resume];

监听网络状态

  • AFNetWorking 也可以用来监听网络状态,当网络状态发生改变的时候会自动调用以下 Block 回调,具体方法如下。
/* 监听网络状态 */
AFNetworkReachabilityManager *manager = [AFNetworkReachabilityManager sharedManager];
[manager setReachabilityStatusChangeBlock:^(AFNetworkReachabilityStatus status) {
    if (status == AFNetworkReachabilityStatusUnknown) {
        NSLog(@"当前网络:未知网络");
    } else if (status == AFNetworkReachabilityStatusNotReachable) {
        NSLog(@"当前网络:没有网络");
    } else if (status == AFNetworkReachabilityStatusReachableViaWWAN) {
        NSLog(@"当前网络:手机流量");
    } else if (status == AFNetworkReachabilityStatusReachableViaWiFi) {
        NSLog(@"当前网络:WiFi");
    }
}];
[manager startMonitoring];

更换头像遇到的问题

在写项目时,遇到需要上传头像,更换头像的时候。后台接受到的图片大小是有一定限制的,因此我们在进行图片上传的时候,一定要注意图片的大小,适当的进行压缩处理

///压缩图片
+ (NSData *)imageCompressToData:(UIImage *)image {
  	NSData *data = UIImageJPEGRepresentation(image, 1.0);
  	if (data.length > 300 * 1024) {
    	if (data.length > 1024 * 1024) {			//1M以及以上
      		data = UIImageJPEGRepresentation(image, 0.1);
    	} else if (data.length > 512 * 1024) {			//0.5M-1M
      		data = UIImageJPEGRepresentation(image, 0.5);
    	} else if (data.length > 300 * 1024) {			//0.25M-0.5M
      		data = UIImageJPEGRepresentation(image, 0.9);
    	}
  	}
  	return data;
}

但是我在压缩到指定文件大小时,虽然会向指定的大小进行压缩,但是有时还是高于指定大小,无法始终控制在指定大小以下。