有的程序员老了,还没听过NSURLSession
有的程序员还嫩,没用过NSURLConnection
有的程序员很单纯,他只知道AFN.程序员
NSURLConnection在iOS9被宣布弃用,NSURLSession从13年发展到如今,终于迎来了它独步江湖的时代.NSURLSession是苹果在iOS7后为HTTP数据传输提供的一系列接口,比NSURLConnection强大,坑少,好用.今天从使用的角度介绍下.缓存
除了NSURLSession
,文中还会频繁地出现NSURLSessionConfiguration
和NSURLSessionTask
两个类.先认识一下,混个脸熟吧.安全
使用NSURLSession,拢共分两步:服务器
既然两步里面都出现了task,就先说说它吧.
NSURLSessionTask能够简单理解为任务:如数据请求任务,下载任务,上传任务and so on.咱们使用的是他的子类们:网络
从这几个子类的名字就能够大概猜出他们的做用了.接下来咱们就从不一样类型的任务出发,来使用session.session
字面上看是和数据相关的任务,但其实dataTask彻底能够胜任downloadTask和uploadTask的工做.这可能也是咱们使用最多的task种类.ide
若是请求的数据比较简单,也不须要对返回的数据作一些复杂的操做.那么咱们可使用带blockui
// 快捷方式得到session对象 NSURLSession *session = [NSURLSession sharedSession]; NSURL *url = [NSURL URLWithString:@"http://www.daka.com/login?username=daka&pwd=123"]; // 经过URL初始化task,在block内部能够直接对返回的数据进行处理 NSURLSessionTask *task = [session dataTaskWithURL:url completionHandler:^(NSData *data, NSURLResponse *response, NSError error) { NSLog(@"%@", [NSJSONSerialization JSONObjectWithData:data options:kNilOptions error:nil]); }]; // 启动任务 [task resume];
POST和GET的区别就在于request,因此使用session的POST请求和GET过程是同样的,区别就在于对request的处理.url
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];
NSURLSession提供了block方式处理返回数据的简便方式,但若是想要在接收数据过程当中作进一步的处理,仍然能够调用相关的协议方法.NSURLSession的代理方法和NSURLConnection有些相似,都是分为接收响应、接收数据、请求完成几个阶段.spa
// 使用代理方法须要设置代理,可是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 { // 请求完成,成功或者失败的处理 }
关键点在代码注释里面都有说起,重要的地方再强调一下:
sessionWithConfiguration:delegate:delegateQueue:
其中:
[NSURLSessionConfiguration defaultSessionConfiguration]
就好(后面会说下这个配置是干吗用的);completionHandler(NSURLSessionResponseAllow);
,才会继续接收服务器返回的数据,进入后面的代理方法.值得一提的是,若是在接收响应的时候须要对返回的参数进行处理(如获取响应头信息等),那么这些处理应该放在前面容许操做的前面.文件下载可使用NSURLSessionDownloadTask这个子类.
NSURLSessionDownloadTask一样提供了经过NSURL和NSURLRequest两种方式来初始化并经过block进行回调的方法.下面以NSURL初始化为例:
SURLSession *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];
response.suggestedFilename
是从相应中取出文件在服务器上存储路径的最后部分,如数据在服务器的url为http://www.daka.com/resources/image/icon.png
, 那么其suggestedFilename就是icon.png.一样的,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 { }
在NSURLSession中,文件上传方式主要有如下两种:
NSURLSessionUploadTask *task = [[NSURLSession sharedSession] uploadTaskWithRequest:request fromFile:fileName completionHandler:^(NSData *data, NSURLResponse *response, NSError *error) { }];
和
[self.session 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协议规定格式的大长串).由于你有90%的可能性用了AFNetworking,即便是本身写的应该也是copy,因此代码就不贴了咱们只说方法呵呵哒.
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];
以目前我对NSURLSession的理解这种断点下载只支持应用内断点,若是程序在下载过程当中途关闭,则不能恢复下载.(暂时对NSURLSession理解还不全面,不敢妄下断论,若有不妥简友们能够沟通下)
此外,task们自身有都拥有下面几个方法
- (void)suspend; - (void)resume; - (void)cancel;
suspend可让当前的任务暂停
resume方法不只能够启动任务,还能够唤醒suspend状态的任务
cancel方法能够取消当前的任务,你也能够向处于suspend状态的任务发送cancel消息,任务若是被取消便不能再恢复到以前的状态.
简单地说,就是session的配置信息.如:
NSURLSessionConfiguration *config = [NSURLSessionConfiguration defaultSessionConfiguration]; // 超时时间 config.timeoutIntervalForRequest = 10; // 是否容许使用蜂窝网络(后台传输不适用) config.allowsCellularAccess = YES; // 还有不少能够设置的属性
有没有发现咱们使用的Configuration都是默认配置:[NSURLSessionConfiguration defaultSessionConfiguration]
,其实它的配置有三种类型:
+ (NSURLSessionConfiguration *)defaultSessionConfiguration; + (NSURLSessionConfiguration *)ephemeralSessionConfiguration; + (NSURLSessionConfiguration *)backgroundSessionConfigurationWithIdentifier:(NSString *)identifier
表示了NSURLSession几种不一样的工做模式.
默认的配置会将缓存存储在磁盘上,第二种瞬时会话模式不会建立持久性存储的缓存,第三种后台会话模式容许程序在后台进行上传下载工做.
除了支持任务的暂停和断点续传,我以为NSURLSession之于NSURLConnection的最伟大的进步就是支持后台上传下载任务,这又是一个能够深刻讨论的话题.但在这方面我尚未进行深刻的研究,待后续了解以后另行开贴.
PS:AFNetWorking从2.0版本就有了基于NSURLSession的系列封装,感兴趣的童鞋自行前往了解.