开发只懂 AFN ?搞定 NSURLSession 才是硬道理
由于傲娇的苹果在 iOS9 之后已经放弃了 NSURLConnection,所以在现在的实际开发中,除了大家常见的 AFN 框架,一般使用的是 iOS7 之后推出的 NSURLSession,作为一名 iOS 开发人员,如果你只知道 AFN 框架来进行网络请求,那就只能说是 too young too simple,sometimes naive。
目录
- NSURLSession 的优势
- NSURLSessionTask 的子类
- NSURLSessionDataTask 发送 GET 请求
- NSURLSessionDataTask 发送 POST 请求
- NSURLSessionDataTask 设置代理发送请求
- 设置代理之后的强引用问题
- NSURLSessionDataTask 简单下载
- NSURLSessionDownloadTask 简单下载
- dataTask 和 downloadTask 下载对比
- 写在最后
- 【补充】NSURLSession 详解离线断点下载的实现
<h4 id='1'> NSURLSession 的优势 </h4>
- NSURLSession 支持 http2.0 协议
- 在处理下载任务的时候可以直接把数据下载到磁盘
- 支持后台下载|上传
- 同一个 session 发送多个请求,只需要建立一次连接(复用了TCP)
- 提供了全局的 session 并且可以统一配置,使用更加方便
- 下载的时候是多线程异步处理,效率更高
<h4 id='2'> NSURLSessionTask 的子类 </h4>
- NSURLSessionTask 是一个抽象类,如果要使用那么只能使用它的子类
- NSURLSessionTask 有两个子类
- NSURLSessionDataTask,可以用来处理一般的网络请求,如 GET | POST 请求等
- NSURLSessionDataTask 有一个子类为 NSURLSessionUploadTask,用于处理上传请求的时候有优势
-
NSURLSessionDownloadTask,主要用于处理下载请求,有很大的优势
NSURLSession 的子类
- NSURLSessionDataTask,可以用来处理一般的网络请求,如 GET | POST 请求等
<h4 id='3'> NSURLSessionDataTask 发送 GET 请求 </h4>
发送 GET 请求的步骤非常简单,只需要两步就可以完成:
-
使用 NSURLSession 对象创建 Task
-
执行 Task
//确定请求路径
NSURL *url = [NSURL URLWithString:@"http://120.25.226.186:32812/login?username=520&pwd=520&type=JSON"];
//创建 NSURLSession 对象
NSURLSession *session = [NSURLSession sharedSession];
/**
根据对象创建 Task 请求
url 方法内部会自动将 URL 包装成一个请求对象(默认是 GET 请求)
completionHandler 完成之后的回调(成功或失败)
param data 返回的数据(响应体)
param response 响应头
param error 错误信息
*/
NSURLSessionDataTask *dataTask = [session dataTaskWithURL:url completionHandler:
^(NSData * _Nullable data, NSURLResponse * _Nullable response, NSError * _Nullable error) {
//解析服务器返回的数据
NSLog(@"%@", [[NSString alloc] initWithData:data encoding:NSUTF8StringEncoding]);
//默认在子线程中解析数据
NSLog(@"%@", [NSThread currentThread]);
}];
//发送请求(执行Task)
[dataTask resume];
<h4 id='4'> NSURLSessionDataTask 发送 POST 请求 </h4>
发送 POST 请求的步骤与发送 GET 请求一样:
1. 使用 NSURLSession 对象创建 Task
2. 执行 Task
```objective-c
//确定请求路径
NSURL *url = [NSURL URLWithString:@"http://120.25.226.186:32812/login"];
//创建可变请求对象
NSMutableURLRequest *requestM = [NSMutableURLRequest requestWithURL:url];
//修改请求方法
requestM.HTTPMethod = @"POST";
//设置请求体
requestM.HTTPBody = [@"username=520&pwd=520&type=JSON" dataUsingEncoding:NSUTF8StringEncoding];
//创建会话对象
NSURLSession *session = [NSURLSession sharedSession];
//创建请求 Task
NSURLSessionDataTask *dataTask = [session dataTaskWithRequest:requestM completionHandler:
^(NSData * _Nullable data, NSURLResponse * _Nullable response, NSError * _Nullable error) {
//解析返回的数据
NSLog(@"%@", [[NSString alloc] initWithData:data encoding:NSUTF8StringEncoding]);
}];
//发送请求
[dataTask resume];
<h4 id='5'> NSURLSessionDataTask 设置代理发送请求 </h4>
-
创建 NSURLSession 对象设置代理
-
使用 NSURLSession 对象创建 Task
-
执行 Task
//确定请求路径
NSURL url = [NSURL URLWithString:@"http://120.25.226.186:32812/login"];
//创建可变请求对象
NSMutableURLRequest requestM = [NSMutableURLRequest requestWithURL:url];
//设置请求方法
requestM.HTTPMethod = @"POST";
//设置请求体
requestM.HTTPBody = [@"username=520&pwd=520&type=JSON" dataUsingEncoding:NSUTF8StringEncoding];
//创建会话对象,设置代理
/
第一个参数:配置信息
第二个参数:设置代理
第三个参数:队列,如果该参数传递nil 那么默认在子线程中执行
*/
NSURLSession *session = [NSURLSession sessionWithConfiguration:[NSURLSessionConfiguration defaultSessionConfiguration]
delegate:self delegateQueue:nil];
//创建请求 Task
NSURLSessionDataTask *dataTask = [session dataTaskWithRequest:requestM];
//发送请求
[dataTask resume];
4. 遵守协议,实现代理方法(常用的有三种代理方法)
```objective-c
-(void)URLSession:(NSURLSession *)session dataTask:(nonnull NSURLSessionDataTask *)dataTask
didReceiveResponse:(nonnull NSURLResponse *)response
completionHandler:(nonnull void (^)(NSURLSessionResponseDisposition))completionHandler {
//子线程中执行
NSLog(@"接收到服务器响应的时候调用 -- %@", [NSThread currentThread]);
self.dataM = [NSMutableData data];
//默认情况下不接收数据
//必须告诉系统是否接收服务器返回的数据
completionHandler(NSURLSessionResponseAllow);
}
-(void)URLSession:(NSURLSession *)session dataTask:(NSURLSessionDataTask *)dataTask didReceiveData:(NSData *)data {
NSLog(@"接受到服务器返回数据的时候调用,可能被调用多次");
//拼接服务器返回的数据
[self.dataM appendData:data];
}
-(void)URLSession:(NSURLSession *)session task:(NSURLSessionTask *)task didCompleteWithError:(NSError *)error {
NSLog(@"请求完成或者是失败的时候调用");
//解析服务器返回数据
NSLog(@"%@", [[NSString alloc] initWithData:self.dataM encoding:NSUTF8StringEncoding]);
}
<h4 id='6'> 设置代理之后的强引用问题 </h4>
- NSURLSession 对象在使用的时候,如果设置了代理,那么 session 会对代理对象保持一个强引用,在合适的时候应该主动进行释放
- 可以在控制器调用 viewDidDisappear 方法的时候来进行处理,可以通过调用 invalidateAndCancel 方法或者是 finishTasksAndInvalidate 方法来释放对代理对象的强引用
-
invalidateAndCancel 方法直接取消请求然后释放代理对象
-
finishTasksAndInvalidate 方法等请求完成之后释放代理对象。
-
<h4 id='7'> NSURLSessionDataTask 简单下载 </h4>
在前面请求数据的时候就相当于一个简单的下载过程,NSURLSessionDataTask 下载文件具体的步骤与上类似:
-
使用 NSURLSession 对象创建一个 Task 请求
-
执行请求
[[[NSURLSession sharedSession] dataTaskWithURL:[NSURL URLWithString:
@"http://120.25.226.186:32812/resources/images/minion_01.png"]
completionHandler:^(NSData * _Nullable data, NSURLResponse * _Nullable response, NSError * _Nullable error) {
//解析数据
UIImage *image = [UIImage imageWithData:data];
//回到主线程设置图片
dispatch_async(dispatch_get_main_queue(), ^{
self.imageView.image = image;
});
}] resume];
<h4 id='8'> NSURLSessionDownloadTask 简单下载 </h4>
1. 使用 NSURLSession 对象创建下载请求
2. 在下载请求中移动文件到指定位置
3. 执行请求
```objective-c
//确定请求路径
NSURL *url = [NSURL URLWithString:@"http://120.25.226.186:32812/resources/images/minion_02.png"];
//创建请求对象
NSURLRequest *request = [NSURLRequest requestWithURL:url];
//创建会话对象
NSURLSession *session = [NSURLSession sharedSession];
//创建会话请求
//优点:该方法内部已经完成了边接收数据边写沙盒的操作,解决了内存飙升的问题
NSURLSessionDownloadTask *downTask = [session downloadTaskWithRequest:request
completionHandler:^(NSURL * _Nullable location, NSURLResponse * _Nullable response, NSError * _Nullable error) {
//默认存储到临时文件夹 tmp 中,需要剪切文件到 cache
NSLog(@"%@", location);//目标位置
NSString *fullPath = [[NSSearchPathForDirectoriesInDomains(NSCachesDirectory, NSUserDomainMask, YES) lastObject]
stringByAppendingPathComponent:response.suggestedFilename];
/**
fileURLWithPath:有协议头
URLWithString:无协议头
*/
[[NSFileManager defaultManager] moveItemAtURL:location toURL:[NSURL fileURLWithPath:fullPath] error:nil];
}];
//发送请求
[downTask resume];
以上方法无法监听下载进度,如要获取下载进度,可以使用代理的方式进行下载。
<h4 id='9'> dataTask 和 downloadTask 下载对比 </h4>
- NSURLSessionDataTask
- 下载文件可以实现离线断点下载,但是代码相对复杂
- NSURLSessionDownloadTask
- 下载文件可以实现断点下载,但不能离线断点下载
- 内部已经完成了边接收数据边写入沙盒的操作
- 解决了下载大文件时的内存飙升问题
<h4 id='10'> 写在最后 </h4>
关于使用 NSURLSession 进行上传文件操作,我只想说真的很麻烦,建议大家时间充沛且有兴趣的可以研究一下,如果不想研究也是可以的,继续使用我们伟大的 AFN 框架就好。至于 AFN 框架的使用,这里就不赘述了,后期如果有时间会更新一些常用的 AFN 使用方法,敬请期待。
附:使用 NSURLSession 上传文件主要步骤及注意点
- 主要步骤:
- 确定上传请求的路径( NSURL )
- 创建可变的请求对象( NSMutableURLRequest )
- 修改请求方法为 POST
- 设置请求头信息(告知服务器端这是一个文件上传请求)
- 按照固定的格式拼接要上传的文件等参数
- 根据请求对象创建会话对象( NSURLSession 对象)
- 根据 session 对象来创建一个 uploadTask 上传请求任务
- 执行该上传请求任务(调用 resume 方法)
- 得到服务器返回的数据,解析数据(上传成功 | 上传失败)
- 注意点:
- 创建可变的请求对象,因为需要修改请求方法为 POST,设置请求头信息
- 设置请求头这个步骤可能会被遗漏
- 要处理上传参数的时候,一定要按照固定的格式来进行拼接
- 需要采用合适的方法来获得上传文件的二进制数据类型( MIMEType,获取方式如下)
-
对着该文件发送一个网络请求,接收到该请求响应的时候,可以通过响应头信息中的 MIMEType 属性得到
-
使用通用的二进制数据类型表示任意的二进制数据 application/octet-stream
-
调用 C 语言的 API 来获取