目录
NSURLSession 的优势
NSURLSessionTask 的子类
NSURLSessionDataTask 发送 GET 请求
NSURLSessionDataTask 发送 POST 请求
NSURLSessionDataTask 设置代理发送请求
设置代理之后的强引用问题
NSURLSessionDataTask 简单下载
NSURLSessionDownloadTask 简单下载
dataTask 和 downloadTask 下载对比
写在最后
【补充】NSURLSession 详解离线断点下载的实现
NSURLSession 的优势
NSURLSession 支持 http2.0 协议
在处理下载任务的时候可以直接把数据下载到磁盘
支持后台下载|上传
同一个 session 发送多个请求,只需要建立一次连接(复用了TCP)
提供了全局的 session 并且可以统一配置,使用更加方便
下载的时候是多线程异步处理,效率更高
NSURLSessionTask 的子类
NSURLSessionTask 是一个抽象类,如果要使用那么只能使用它的子类
NSURLSessionTask 有两个子类
NSURLSessionDataTask,可以用来处理一般的网络请求,如 GET | POST 请求等
NSURLSessionDataTask 有一个子类为 NSURLSessionUploadTask,用于处理上传请求的时候有优势
NSURLSessionDownloadTask,主要用于处理下载请求,有很大的优势
NSURLSession 的子类
NSURLSessionDataTask 发送 GET 请求
发送 GET 请求的步骤非常简单,只需要两步就可以完成:
使用 NSURLSession 对象创建 Task
执行 Task
//确定请求路径NSURL*url = [NSURLURLWithString:@"http://120.25.226.186:32812/login?username=520&pwd=520&type=JSON"];//创建 NSURLSession 对象NSURLSession*session = [NSURLSessionsharedSession];/**
根据对象创建 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(@"%@", [[NSStringalloc] initWithData:data encoding:NSUTF8StringEncoding]);//默认在子线程中解析数据NSLog(@"%@", [NSThreadcurrentThread]); }];//发送请求(执行Task)[dataTask resume];
NSURLSessionDataTask 发送 POST 请求
发送 POST 请求的步骤与发送 GET 请求一样:
使用 NSURLSession 对象创建 Task
执行 Task
//确定请求路径NSURL*url = [NSURLURLWithString:@"http://120.25.226.186:32812/login"];//创建可变请求对象NSMutableURLRequest*requestM = [NSMutableURLRequestrequestWithURL:url];//修改请求方法requestM.HTTPMethod=@"POST";//设置请求体requestM.HTTPBody= [@"username=520&pwd=520&type=JSON"dataUsingEncoding:NSUTF8StringEncoding];//创建会话对象NSURLSession*session = [NSURLSessionsharedSession];//创建请求 TaskNSURLSessionDataTask*dataTask = [session dataTaskWithRequest:requestM completionHandler: ^(NSData* _Nullable data,NSURLResponse* _Nullable response,NSError* _Nullable error) {//解析返回的数据NSLog(@"%@", [[NSStringalloc] initWithData:data encoding:NSUTF8StringEncoding]); }];//发送请求[dataTask resume];
NSURLSessionDataTask 设置代理发送请求
创建 NSURLSession 对象设置代理
使用 NSURLSession 对象创建 Task
执行 Task
//确定请求路径NSURL*url = [NSURLURLWithString:@"http://120.25.226.186:32812/login"];//创建可变请求对象NSMutableURLRequest*requestM = [NSMutableURLRequestrequestWithURL:url];//设置请求方法requestM.HTTPMethod=@"POST";//设置请求体requestM.HTTPBody= [@"username=520&pwd=520&type=JSON"dataUsingEncoding:NSUTF8StringEncoding];//创建会话对象,设置代理/**
第一个参数:配置信息
第二个参数:设置代理
第三个参数:队列,如果该参数传递nil 那么默认在子线程中执行
*/NSURLSession*session = [NSURLSessionsessionWithConfiguration:[NSURLSessionConfigurationdefaultSessionConfiguration] delegate:selfdelegateQueue:nil];//创建请求 TaskNSURLSessionDataTask*dataTask = [session dataTaskWithRequest:requestM];//发送请求[dataTask resume];
遵守协议,实现代理方法(常用的有三种代理方法)
-(void)URLSession:(NSURLSession*)session dataTask:(nonnullNSURLSessionDataTask*)dataTask didReceiveResponse:(nonnullNSURLResponse*)response completionHandler:(nonnullvoid(^)(NSURLSessionResponseDisposition))completionHandler {//子线程中执行NSLog(@"接收到服务器响应的时候调用 -- %@", [NSThreadcurrentThread]);self.dataM= [NSMutableDatadata];//默认情况下不接收数据//必须告诉系统是否接收服务器返回的数据completionHandler(NSURLSessionResponseAllow);}-(void)URLSession:(NSURLSession*)session dataTask:(NSURLSessionDataTask*)dataTask didReceiveData:(NSData*)data {NSLog(@"接受到服务器返回数据的时候调用,可能被调用多次");//拼接服务器返回的数据[self.dataMappendData:data];}-(void)URLSession:(NSURLSession*)session task:(NSURLSessionTask*)task didCompleteWithError:(NSError*)error {NSLog(@"请求完成或者是失败的时候调用");//解析服务器返回数据NSLog(@"%@", [[NSStringalloc] initWithData:self.dataMencoding:NSUTF8StringEncoding]);}
设置代理之后的强引用问题
NSURLSession 对象在使用的时候,如果设置了代理,那么 session 会对代理对象保持一个强引用,在合适的时候应该主动进行释放
可以在控制器调用 viewDidDisappear 方法的时候来进行处理,可以通过调用 invalidateAndCancel 方法或者是 finishTasksAndInvalidate 方法来释放对代理对象的强引用
invalidateAndCancel 方法直接取消请求然后释放代理对象
finishTasksAndInvalidate 方法等请求完成之后释放代理对象。
[self.sessionfinishTasksAndInvalidate];
NSURLSessionDataTask 简单下载
在前面请求数据的时候就相当于一个简单的下载过程,NSURLSessionDataTask 下载文件具体的步骤与上类似:
使用 NSURLSession 对象创建一个 Task 请求
执行请求
[[[NSURLSessionsharedSession] dataTaskWithURL:[NSURLURLWithString:@"http://120.25.226.186:32812/resources/images/minion_01.png"] completionHandler:^(NSData* _Nullable data,NSURLResponse* _Nullable response,NSError* _Nullable error) {//解析数据UIImage*image = [UIImageimageWithData:data];//回到主线程设置图片dispatch_async(dispatch_get_main_queue(), ^{self.imageView.image= image; }); }] resume];
NSURLSessionDownloadTask 简单下载
使用 NSURLSession 对象创建下载请求
在下载请求中移动文件到指定位置
执行请求
//确定请求路径NSURL*url = [NSURLURLWithString:@"http://120.25.226.186:32812/resources/images/minion_02.png"];//创建请求对象NSURLRequest*request = [NSURLRequestrequestWithURL:url];//创建会话对象NSURLSession*session = [NSURLSessionsharedSession];//创建会话请求//优点:该方法内部已经完成了边接收数据边写沙盒的操作,解决了内存飙升的问题NSURLSessionDownloadTask*downTask = [session downloadTaskWithRequest:request completionHandler:^(NSURL* _Nullable location,NSURLResponse* _Nullable response,NSError* _Nullable error) {//默认存储到临时文件夹 tmp 中,需要剪切文件到 cacheNSLog(@"%@", location);//目标位置NSString*fullPath = [[NSSearchPathForDirectoriesInDomains(NSCachesDirectory,NSUserDomainMask,YES) lastObject] stringByAppendingPathComponent:response.suggestedFilename];/**
fileURLWithPath:有协议头
URLWithString:无协议头
*/[[NSFileManagerdefaultManager] moveItemAtURL:location toURL:[NSURLfileURLWithPath:fullPath] error:nil]; }];//发送请求[downTask resume];
以上方法无法监听下载进度,如要获取下载进度,可以使用代理的方式进行下载。
dataTask 和 downloadTask 下载对比
NSURLSessionDataTask
下载文件可以实现离线断点下载,但是代码相对复杂
NSURLSessionDownloadTask
下载文件可以实现断点下载,但不能离线断点下载
内部已经完成了边接收数据边写入沙盒的操作
解决了下载大文件时的内存飙升问题
写在最后
关于使用 NSURLSession 进行上传文件操作,我只想说真的很麻烦,建议大家时间充沛且有兴趣的可以研究一下,如果不想研究也是可以的,继续使用我们伟大的 AFN 框架就好。至于 AFN 框架的使用,这里就不赘述了,后期如果有时间会更新一些常用的 AFN 使用方法,敬请期待。
附:使用 NSURLSession 上传文件主要步骤及注意点
主要步骤:
确定上传请求的路径( NSURL )
创建可变的请求对象( NSMutableURLRequest )
修改请求方法为 POST
设置请求头信息(告知服务器端这是一个文件上传请求)
按照固定的格式拼接要上传的文件等参数
根据请求对象创建会话对象( NSURLSession 对象)
根据 session 对象来创建一个 uploadTask 上传请求任务
执行该上传请求任务(调用 resume 方法)
得到服务器返回的数据,解析数据(上传成功 | 上传失败)
注意点:
创建可变的请求对象,因为需要修改请求方法为 POST,设置请求头信息
设置请求头这个步骤可能会被遗漏
要处理上传参数的时候,一定要按照固定的格式来进行拼接
需要采用合适的方法来获得上传文件的二进制数据类型( MIMEType,获取方式如下)
点击这里搜索
对着该文件发送一个网络请求,接收到该请求响应的时候,可以通过响应头信息中的 MIMEType 属性得到
使用通用的二进制数据类型表示任意的二进制数据 application/octet-stream
调用 C 语言的 API 来获取
[self mimeTypeForFileAtPath:@"此处为上传文件的路径"]
文/Mortal_Master(作者)
原文链接:http://www.jianshu.com/p/b0ddadd34037
著作权归作者所有,转载请联系作者获得授权,并标注“作者”。