NSURLSession 概述

2017-09-12  本文已影响34人  林大鹏

主要参考:
使用NSURLSession
开发只懂 AFN ?搞定 NSURLSession 才是硬道理

想了解AFNetworking,需要先了解下NSURLSession。

NSURLSession 的优势

NSURLSessionTask (抽象类)

使用NSURLSession,共分两步:

这里的任务就是NSURLSessionTaskNSURLSessionTask 是一个抽象类,如果要使用那么只能使用它的子类

NSURLSessionTask有两个子类:

NSURLSession的子类

image.png

NSURLSessionDataTask

这是一个和数据相关的任务,但其实dataTask也可以胜任downloadTaskuploadTask的工作,这一般是我们使用最多的task种类。

简单GET请求:
如果请求数据比较简单,也不需要对返回的数据做一些复杂的操作,我们可以使用带blockapi

//确定请求路径
 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];

注意:所有类型的task都要调用resume方法才会开始进行请求。

简单POST请求
POSTGET的区别就在于request,所以使用sessionPOST请求和GET过程是一样的,区别就在于对request的处理。

NSURL *url = [NSURL URLWithString:@"http://www.daka.com/login"];
NSMutableURLRequest *request = [NSMutableURLRequest requestWithURL:url];
request.HTTPMethod = @"POST";
request.HTTPBody = [@"username=daka&pwd=123" dataUsingEncoding:NSUTF8StringEncoding];

NSURLSession *session = [NSURLSession sharedSession];
// 由于要先对request先行处理,我们通过request初始化task
NSURLSessionTask *task = [session dataTaskWithRequest:request
                               completionHandler:^(NSData *data, NSURLResponse *response, NSError *error) { NSLog(@"%@", [NSJSONSerialization JSONObjectWithData:data options:kNilOptions error:nil]); }];
[task resume];

NSURLSessionDataDelegate代理方法

NSURLSession 提供了block方式处理返回的数据的简便方式,但如果想要在接收数据过程中做进一步的处理,仍然可以调用相关的协议方法:NSURLSession的代理方法和NSURLConnection类似,都是分为接收响应、接收数据、请求完成几个阶段。

// 使用代理方法需要设置代理,但是session的delegate属性是只读的,要想设置代理只能通过这种方式创建session
NSURLSession *session = [NSURLSession sessionWithConfiguration:[NSURLSessionConfiguration defaultSessionConfiguration]
                                                  delegate:self
                                             delegateQueue:[[NSOperationQueue alloc] init]];

// 创建任务(因为要使用代理方法,就不需要block方式的初始化了)
NSURLSessionDataTask *task = [session dataTaskWithRequest:[NSURLRequest requestWithURL:[NSURL URLWithString:@"http://www.daka.com/login?userName=daka&pwd=123"]]];

// 启动任务
[task resume];

//对应的代理方法如下:

// 1.接收到服务器的响应
- (void)URLSession:(NSURLSession *)session dataTask:(NSURLSessionDataTask *)dataTask didReceiveResponse:(NSURLResponse *)response completionHandler:(void (^)(NSURLSessionResponseDisposition))completionHandler {
    // 允许处理服务器的响应,才会继续接收服务器返回的数据
    completionHandler(NSURLSessionResponseAllow);
}

// 2.接收到服务器的数据(可能调用多次)
- (void)URLSession:(NSURLSession *)session dataTask:(NSURLSessionDataTask *)dataTask didReceiveData:(NSData *)data {
    // 处理每次接收的数据
}

// 3.请求成功或者失败(如果失败,error有值)
- (void)URLSession:(NSURLSession *)session task:(NSURLSessionTask *)task didCompleteWithError:(NSError *)error {
    // 请求完成,成功或者失败的处理
}

注意点:

NSURLSessionDownloadTask

文件下载可以使用NSURLSessionDownloadTask这个子类.

简单下载
NSURLSessionDownloadTask同样提供了通过NSURLNSURLRequest两种方式来初始化并通过block来进行回调的方法。下面以NSURL初始化为例:

NSURLSession *session = [NSURLSession sharedSession];
NSURL *url = [NSURL URLWithString:@"http://www.daka.com/resources/image/icon.png"] ;
NSURLSessionDownloadTask *task = [session downloadTaskWithURL:url completionHandler:^(NSURL *location, NSURLResponse *response, NSError *error) {
    // location是沙盒中tmp文件夹下的一个临时url,文件下载后会存到这个位置,由于tmp中的文件随时可能被删除,所以我们需要自己需要把下载的文件挪到需要的地方
    NSString *path = [[NSSearchPathForDirectoriesInDomains(NSCachesDirectory, NSUserDomainMask, YES) lastObject] stringByAppendingPathComponent:response.suggestedFilename];
    // 剪切文件
    [[NSFileManager defaultManager] moveItemAtURL:location toURL:[NSURL fileURLWithPath:path] error:nil];
}];
 // 启动任务
[task resume];

注意点:

NSURLSessionDownloadDelegate代理方法

同样的,downloadTask也提供了配套的代理方法

// 每次写入调用(会调用多次)
- (void)URLSession:(NSURLSession *)session downloadTask:(NSURLSessionDownloadTask *)downloadTask didWriteData:(int64_t)bytesWritten totalBytesWritten:(int64_t)totalBytesWritten totalBytesExpectedToWrite:(int64_t)totalBytesExpectedToWrite {
    // 可在这里通过已写入的长度和总长度算出下载进度
    CGFloat progress = 1.0 * totalBytesWritten / totalBytesExpectedToWrite; NSLog(@"%f",progress);
}

// 下载完成调用
- (void)URLSession:(NSURLSession *)session
  downloadTask:(NSURLSessionDownloadTask *)downloadTask
  didFinishDownloadingToURL:(NSURL *)location {
    // location还是一个临时路径,需要自己挪到需要的路径(caches下面)
    NSString *filePath = [[NSSearchPathForDirectoriesInDomains(NSCachesDirectory, NSUserDomainMask, YES) lastObject] stringByAppendingPathComponent:downloadTask.response.suggestedFilename];
    [[NSFileManager defaultManager] moveItemAtURL:location toURL:[NSURL fileURLWithPath:filePath] error:nil];
}

// 任务完成调用
- (void)URLSession:(NSURLSession *)session task:(NSURLSessionTask *)task didCompleteWithError:(NSError *)error {

}

NSURLSessionUploadTask

NSURLSession中,文件上传方式主要有以下两种:

NSURLSessionUploadTask *task = [[NSURLSession sharedSession] uploadTaskWithRequest:request
                                       fromFile:fileName
                              completionHandler:^(NSData *data, NSURLResponse *response, NSError *error) {
}];

NSURLSessionUploadTask *task = [[NSURLSession sharedSession] uploadTaskWithRequest:request
                        fromData:body
               completionHandler:^(NSData *data, NSURLResponse *response, NSError *error) {
 NSLog(@"-------%@", [NSJSONSerialization JSONObjectWithData:data options:kNilOptions error:nil]);
 }];

出于安全性考虑,通常我们会使用POST方式进行文件上传,所以较多使用第二种方式。

但是NSURLSession并没有为我们提供比NSURLConnection更方便的上传方式,方法中body处的参数需要填写request请求体(http协议规定格式的字符串)。

断点下载

NSURLSessionDownloadTask提供了与断点下载相关的几个方法:

// 使用这种方式取消下载可以得到将来用来恢复的数据,保存起来
[self.task cancelByProducingResumeData:^(NSData *resumeData) {
    self.resumeData = resumeData;
}];

// 由于下载失败导致的下载中断会进入此协议方法,也可以得到用来恢复的数据
- (void)URLSession:(NSURLSession *)session task:(NSURLSessionTask *)task didCompleteWithError:(NSError *)error
{
    // 保存恢复数据
    self.resumeData = error.userInfo[NSURLSessionDownloadTaskResumeData];
}

// 恢复下载时接过保存的恢复数据
self.task = [self.session downloadTaskWithResumeData:self.resumeData];
// 启动任务
[self.task resume];

其它

此外,task自身还拥有这几个方法:

- (void)suspend;
- (void)resume;
- (void)cancel;

suspend 可以让当前的任务暂停
resume方法不仅可以启动任务,还可以唤醒suspend状态的任务。
cancel 方法可以取消当前的任务,也可以向处于suspend状态的任务发送cancel消息,任务如果被取消便不能在恢复到之前的状态。

NSURLSessionConfiguration

NSURLSession的配置信息,如:

NSURLSessionConfiguration *config = [NSURLSessionConfiguration defaultSessionConfiguration];
// 超时时间
config.timeoutIntervalForRequest = 10;
// 是否允许使用蜂窝网络(后台传输不适用)
config.allowsCellularAccess = YES;
// 还有很多可以设置的属性

一般都是使用默认配置,但其实它的配置有三种类型:

// 将缓存存储在磁盘上
+ (NSURLSessionConfiguration *)defaultSessionConfiguration;
// 瞬时会话模式不会创建持久性存储缓存
+ (NSURLSessionConfiguration *)ephemeralSessionConfiguration;
// 后台会话模式,允许程序在后台进行上传下载工作
+ (NSURLSessionConfiguration *)backgroundSessionConfigurationWithIdentifier:(NSString *)identifier

注意点

1. 设置代理的强引用问题

2. 使用NSURLSession上传文件主要步骤和注意点

主要步骤:

注意点

上一篇 下一篇

猜你喜欢

热点阅读