文章目录

  • 一、前文
  • 二、NSURLSession与NSURLConnection区别
  • 2.1 使用现状
  • 2.2 普通任务、上传和下载
  • 2.3 下载任务方式
  • 2.4 请求方法的控制
  • 2.5 断点续传的方式
  • 2.6 配置信息
  • 三、NSURLSession的优势
  • 四、代码部分
  • 4.1 Xcode不支持http访问的解决办法
  • 4.2 手动缓存(繁琐)
  • 4.3 自动缓存
  • 四、 demo


一、前文

写这篇文章,是为了总结一周内的学到的关于NSURLSession的相关知识,并且实现了简单的下载缓存功能,NSURLSession有很多很重要的知识,文章会慢慢同步更新。

二、NSURLSession与NSURLConnection区别

2.1 使用现状

NSURLSession是NSURLConnection的替代者,在2013年的WWDC(苹果全球开发者大会)随着iOS7一起发布,是 对NSURLConnection重构后的网络访问接口。从iOS9.0以后,苹果废弃了NSURLConnection 的两个发送请求的方法(同步请求、异步请求),初始化网络连接(initWithRequest:delegate:)也被设置为过期 。

2.2 普通任务、上传和下载

NSURLSession针对普通、上传和下载分别提供了专门的解决方案,三种网络请求分别是:NSURLSessionDataTaskNSURLSessionUploadTaskNSURLSessionDownloadTask。创建的task都是挂起状态,需要resume才能执行 。

相同的点:

  1. 当服务器返回的数据较小时:NSURLSession与NSURLConnection执行普通任务的操作步骤没有区别。
  2. 执行上传任务时:NSURLSession与NSURLConnection同样需要设置POST请求的请求体进行上传

2.3 下载任务方式

NSURLConnection下载文件时,先将文件下载到内存,然后再将文件写入沙盒,如果文件比较大,会出现内存上升的情况。
NSURLSession下载文件,默认会下载到沙盒中的tem文件夹中,避免了上述问题的出现,但是下载完成后,会将tem文件夹中的临时文件删除,需要在初始化任务方法时,在completionHandler中增加保存文件的代码。

2.4 请求方法的控制

NSURLConnection实例化对象,实例化开始,默认请求就发送(同步发送),不需要调用start方法。而cancel 可以停止请求的发送,停止后不能继续访问,需要创建新的请求。
NSURLSession有三个控制方法,取消(cancel),暂停(suspend),继续(resume),暂停后可以通过继续恢复当前的请求任务。

2.5 断点续传的方式

NSURLConnection进行断点下载,通过设置访问请求的HTTPHeaderField的Range属性,开启运行循环,NSURLConnection的代理方法作为运行循环的事件源,接收到下载数据时代理方法就会持续调用,并使用NSOutputStream管道流进行数据保存。

NSURLSession进行断点下载,当暂停下载任务后,如果downloadTask (下载任务)为非空,调用 cancelByProducingResumeData:(void (^)(NSData *resumeData))completionHandler这个方法,这个方法接收一个参数,完成处理代码块,这个代码块有一个 NSData 参数 resumeData,如果 resumeData 非空,我们就保存这个对象到视图控制器的 resumeData 属性中。在点击再次下载时,通过调用[ [self.session downloadTaskWithResumeData:self.resumeData]resume]方法进行继续下载操作。

经过以上比较可以发现,使用NSURLSession进行断点下载更加便捷。

2.6 配置信息

NSURLSession的构造方法sessionWithConfiguration:delegate:delegateQueue中有一个 NSURLSessionConfiguration类的参数可以设置配置信息,其决定了cookie,安全和高速缓存策略,最大主机连接数,资源管理,网络超时等配置。NSURLConnection不能进行这个配置,相比于 NSURLConnection 依赖于一个全局的配置对象,缺乏灵活性而言,NSURLSession 有很大的改进了。

NSURLSession可以设置三种配置信息,分别通过调用三个累方法返回配置对象:

  1. + (NSURLSessionConfiguration *)defaultSessionConfiguration,配置信息使用基于硬盘的持久话Cache,保存用户的证书到钥匙串,使用共享cookie存储;
  2. + (NSURLSessionConfiguration *)ephemeralSessionConfiguration,配置信息和default大致相同。除了,不会把cache,证书,或者任何和Session相关的数据存储到硬盘,而是存储在内存中,生命周期和Session一致。比如浏览器无痕浏览等功能就可以基于这个来做;
  3. + (NSURLSessionConfiguration *)backgroundSessionConfigurationWithIdentifier:(NSString *)identifier,配置信息可以创建一个可以在后台甚至APP已经关闭的时候仍然在传输数据的session。注意,后台Session一定要在创建的时候赋予一个唯一的identifier,这样在APP下次运行的时候,能够根据identifier来进行相关的区分。如果用户关闭了APP,IOS 系统会关闭所有的background Session。而且,被用户强制关闭了以后,IOS系统不会主动唤醒APP,只有用户下次启动了APP,数据传输才会继续。

三、NSURLSession的优势

  • NSURLSession支持http2.0协议
  • 处理下载任务的时候可以直接把数据下载到磁盘中
  • 支持后台下载和上传
  • 同一个session发送多次请求,只需要建立一次连接(复用了TCP)
  • 提供了全局的session并且可以统一配置,使用更加方便
  • 下载的时候是多线程异步处理,效率更高

四、代码部分

4.1 Xcode不支持http访问的解决办法

Info.plist中添加NSAppTransportSecurity类型Dictionary。
NSAppTransportSecurity下添加NSAllowsArbitraryLoads类型Boolean,值设为YES

4.2 手动缓存(繁琐)

遵守NSURLSessionDelegate,NSURLSessionDownloadDelegate协议。

  1. 声明相关属性
// 下载任务
@property (nonatomic, strong) NSURLSessionDownloadTask *downloadTask;
// 网络会话
@property (nonatomic, strong) NSURLSession *downLoadSession;
// 配置参数
@property (nonatomic, strong) NSURLSessionConfiguration *sessionConfig;
// 数据请求
@property (nonatomic, strong) NSURLRequest *videoRequest;
  1. 配置相关参数并下载
//参数设置类  简单的网络下载使用 defaultSessionConfiguration即可         
self.sessionConfig = [NSURLSessionConfiguration defaultSessionConfiguration];
// 创建网络会话
self.downLoadSession = [NSURLSession sessionWithConfiguration:self.sessionConfig delegate:self delegateQueue:[NSOperationQueue new]];

// 数据请求
self.videoRequest = [NSURLRequest requestWithURL:[NSURL URLWithString:stringURL] cachePolicy:NSURLRequestReturnCacheDataElseLoad timeoutInterval:timeOut];

// 创建 下载任务
self.downloadTask = [self.downLoadSession downloadTaskWithRequest:self.videoRequest];

// 启动下载任务
[self.downloadTask resume];
  1. 代理实现
// 下载过程
- (void)URLSession:(NSURLSession *)session downloadTask:(NSURLSessionDownloadTask *)downloadTask didWriteData:(int64_t)bytesWritten totalBytesWritten:(int64_t)totalBytesWritten totalBytesExpectedToWrite:(int64_t)totalBytesExpectedToWrite{
    self.currentProgress = totalBytesWritten / (double)totalBytesExpectedToWrite;
//    dispatch_sync(dispatch_get_main_queue(), ^{
        NSLog(@"进度%f", self.currentProgress);
//    });
}

// 下载完成,无论成败
- (void)URLSession:(NSURLSession *)session task:(NSURLSessionTask *)task didCompleteWithError:(NSError *)error{
    NSLog(@"function == %s, line == %d, error == %@", __FUNCTION__, __LINE__, error);
}

// 下载完成,获取下载内容
- (void)URLSession:(NSURLSession *)session downloadTask:(NSURLSessionDownloadTask *)downloadTask didFinishDownloadingToURL:(NSURL *)location{
    
    NSString *file = [self.documentsPath stringByAppendingString:self.videoFileName];
    
    [self.videoFilesDownloadedDic setValue:file forKey:self.videoFileName];
    NSLog(@"file%@", file);
    

    NSError *err = nil;
    
    // 删除之前相同的文件
    BOOL remove  = [self.manager removeItemAtPath:file error:nil];
    
    // 移动用move,复制用copy
    BOOL success = [self.manager moveItemAtPath:location.path toPath:file error:&err];
    
    if (success){
        NSLog(@"移动成功");
    } else {
        NSLog(@"失败%@", err);
    }
}

4.3 自动缓存

待补充,还在学习

四、 demo

这里是demo,demo 是一步一步更新,目前只能使用于很小的范围内,但是初步学习NSURLSession我觉得是够了,还待完善。