NSURLSession

NSURLSession 简介
NSURLSession是iOS7中新的网络接口,它与咱们熟悉的NSURLConnection(iOS9以后不推荐使用)是并列的。当程序在前台时,NSURLSession与NSURLConnection可以互为替代工作。注意,如果用户强制将程序关闭,NSURLSession会断掉。

NSURLConnection这个名字,实际上指的是一组构成Foundation框架中URL加载系统的相互关联的组件:NSURLRequest,NSURLResponse,NSURLProtocol,NSURLCache,NSHTTPCookieStorage,NSURLCredentialStorage,以及和它同名的NSURLConnection。

在WWDC 2013中,Apple的团队对NSURLConnection进行了重构,并推出了NSURLSession作为替代。

NSURLSession也是一组相互依赖的类,它的大部分组件和NSURLConnection中的组件相同如NSURLRequest,NSURLCache等。而NSURLSession的不同之处在于,它将NSURLConnection替换为NSURLSession和NSURLSessionConfiguration,以及3个NSURLSessionTask的子类:NSURLSessionDataTask, NSURLSessionUploadTask, 和NSURLSessionDownloadTask。

NSURLSession 优势
后台上传和下载:只需在创建NSURLSession的时候配置一个选项,就能得到后台网络的所有好处。这样可以延长电池寿命,并且还支持UIKit的多task,在进程间使用相同的委托模型。

能够暂停和恢复网络操作:使用NSURLSession API能够暂停,停止,恢复所有的网络任务,再也完全不需要子类化NSOperation。

可配置的容器:对于NSURLSession里面的requests来说,每个NSURLSession都是可配置的容器。举个例来说,假如你需要设置HTTP header选项,你只用做一次,session里面的每个request就会有同样的配置。

提高认证处理:认证是在一个指定的连接基础上完成的。在使用NSURLConnection时,如果发出一个访问,会返回一个任意的request。此时,你就不能确切的知道哪个request收到了访问。而在NSURLSession中,就能用代理处理认证。

丰富的代理模式:在处理认证的时候,NSURLConnection有一些基于异步的block方法,但是它的代理方法就不能处理认证,不管请求是成功或是失败。在NSURLSession中,可以混合使用代理和block方法处理认证。

上传和下载通过文件系统:它鼓励将数据(文件内容)从元数据(URL和settings)中分离出来。

NSURLSessionConfiguration
简介
NSURLSessionConfiguration主要用于配置NSURLSession会话属性,其工作模式主要有三种:

默认模式(defaultSessionConfiguration):使用硬盘来缓存数据,可以使用缓存的Cache,Cookie,鉴权。

及时模式(ephemeralSessionConfiguration): 临时session配置,与默认配置相比,这个配置不会将缓存、cookie等存在本地,只会存在内存里,所以当程序退出时,所有的数据都会消失

后台模式(backgroundSessionConfiguration):在后台完成上传下载,创建Configuration对象的时候需要给一个NSString的ID用于追踪完成工作的Session是哪一个。 做远程push通知或是应用程序挂起的时候就要用到这个configuration。

常用属性
identifie:设置后台任务的标识符

requestCachePolicy:设置缓存策略

timeoutIntervalForRequest:设置请求超时时长

networkServiceType:设置网络服务类型

HTTPAdditionalHeaders:设置头部参数

allowsCellularAccess:设置是否在非无线的情况下请求网络(使用蜂窝数据)

HTTPMaximumConnectionsPerHost:设置主机网络连接个数

discretionary: discretionary属性为YES时表示当程序在后台运作时由系统自己选择最佳的网络连接配置,该属性可以节省通过蜂窝连接的带宽。在使用后台传输数据的时候,建议使用discretionary属性,而不是allowsCellularAccess属性,因为它会把WiFi和电源可用性考虑在内。这个标志允许系统为分配任务进行性能优化。这意味着只有当设备有足够电量时,设备才通过Wifi进行数据传输。如果电量低,或者只仅有一个蜂窝连接,传输任务是不会运行的。后台传输总是在discretionary模式下运行。

NSURLSession是为了代替NSURLConnection而设计的。Sessions的所有工作都是通过NSURLSessionTask对象完成。可以用block,delegate,或者两者混合来创建task。

创建一个NSURLSession,系统提供了三个创建方法:

sharedSession:使用静态的sharedSession方法,该类使用共享的会话,该会话使用全局的Cache,Cookie和证书。

sessionWithConfiguration:粒度较低,就是根据NSURLSessionConfiguration对象创建一个Session,系统默认创建一个新的OperationQueue处理Session的消息。

sessionWithConfiguration:delegate:delegateQueue:粒度较高,可以设定回调的delegate(注意这个回调delegate会被强引用),并且可以设定delegate在哪个OperationQueue回调,如果我们将其设置为[NSOperationQueue mainQueue]就能在主线程进行回调,非常的方便。

注意:二三两种方式可以创建一个新会话并定制其会话类型。该方式中指定了session的委托和委托所处的队列。当不再需要连接时,可以调用Session的invalidateAndCancel直接关闭,或者调用finishTasksAndInvalidate等待当前Task结束后关闭。这时Delegate会收到URLSession:didBecomeInvalidWithError:这个事件。Delegate收到这个事件之后会被解引用。

NSURLSessionTask 会话任务

NSURLSessionTask是一个抽象子类,它有三个子类:NSURLSessionDataTask,NSURLSessionUploadTask和NSURLSessionDownloadTask。这三个类封装了现代应用程序的三个基本网络任务:获取数据(比如JSON或XML),以及上传和下载。其集成关系如下图:

NSURLSessionDataTask
这个task调用HTTP GET请求从服务器获取数据。返回的数据格式是NSData。可能需要你自己转换成XML、JSON等数据格式。
通过request对象或url创建:
// 1、通过request创建

  • (NSURLSessionDataTask *)dataTaskWithRequest:(NSURLRequest *)request;
    // 2、通过url创建
  • (NSURLSessionDataTask *)dataTaskWithURL:(NSURL *)url;
    通过request对象或url创建,同时指定任务完成后通过completionHandler指定回调的代码块:
    // 1、通过request创建
  • (NSURLSessionDataTask *)dataTaskWithRequest:(NSURLRequest *)request completionHandler:(void (^)(NSData * __nullable data, NSURLResponse * __nullable response, NSError * __nullable error))completionHandler;
    // 2、通过url创建
  • (NSURLSessionDataTask *)dataTaskWithURL:(NSURL *)url completionHandler:(void (^)(NSData * __nullable data, NSURLResponse * __nullable response, NSError * __nullable error))completionHandler;

NSURLSessionUploadTask
这个类是上传用的,在数据传输过程中,这个代理方法能观察网络状况。
通过request创建,在上传时指定文件源或数据源。
// 1、通过指定文件源上传

  • (NSURLSessionUploadTask *)uploadTaskWithRequest:(NSURLRequest *)request fromFile:(NSURL *)fileURL;

// 2、通过指定数据源上传

  • (NSURLSessionUploadTask *)uploadTaskWithRequest:(NSURLRequest *)request fromData:(NSData *)bodyData;

// 3、通过request上传

  • (NSURLSessionUploadTask *)uploadTaskWithStreamedRequest:(NSURLRequest *)request;

在创建upload task对象时,通过completionHandler指定任务完成后的回调代码块:

  • (NSURLSessionUploadTask *)uploadTaskWithRequest:(NSURLRequest *)request fromFile:(NSURL *)fileURL completionHandler:(void (^)(NSData *data, NSURLResponse *response, NSError *error))completionHandler;

  • (NSURLSessionUploadTask *)uploadTaskWithRequest:(NSURLRequest *)request fromData:(NSData *)bodyData completionHandler:(void (^)(NSData *data, NSURLResponse *response, NSError *error))completionHandler;
    NSURLSessionDownloadTask

NSURLSessionDownloadTask
NSURLSessionDownloadTask下载文件可以在下载中挂起,恢复。

  • (NSURLSessionDownloadTask *)downloadTaskWithRequest:(NSURLRequest *)request;
  • (NSURLSessionDownloadTask *)downloadTaskWithURL:(NSURL *)url;
  • (NSURLSessionDownloadTask *)downloadTaskWithResumeData:(NSData *)resumeData;
    下载任务支持断点续传,第三种方式是通过之前已经下载的数据来创建下载任务。

同样的可以通过completionHandler指定任务完成后的回调代码块:

// 方法1

  • (NSURLSessionDownloadTask *)downloadTaskWithRequest:(NSURLRequest *)request completionHandler:(void (^)(NSURL *location, NSURLResponse *response, NSError *error))completionHandler;

// 方法2

  • (NSURLSessionDownloadTask *)downloadTaskWithURL:(NSURL *)url completionHandler:(void (^)(NSURL *location, NSURLResponse *response, NSError *error))completionHandler;

// 方法3

  • (NSURLSessionDownloadTask *)downloadTaskWithResumeData:(NSData *)resumeData completionHandler:(void (^)(NSURL *location, NSURLResponse *response, NSError *error))completionHandler;

Tips:

1、task的类型直接写到一个临时文件中。

2、下载过程中会调用URLSession:downloadTask:didWriteData:totalBytesWritten:totalBytesExpectedToWrite: 去更新状态。

3、当task完成,URLSession:downloadTask:didFinishDownloadingToURL:会被调用。此时你可以将临时文件保存到永久文件中。

4、下载失败或是取消还可以得到已经下载的那一部分数据。

NSURLSessionTask 任务挂起

[task resume];
同时管理多个task时,可以用taskIdentifier属性来唯一标识task。

在iOS中,默认只有一个线程,称为主线程,主线程主要处理用户交互和界面刷新,通过NSURLSession请求数据,一般会开辟子线程进行,以防止主线程拥堵,导致界面卡死甚至奔溃。因此在获取到数据之后,刷新界面应该在主线程执行刷新操作。获取主线程刷新界面的方法有多重方式,多线程章节会详细讲解线程创建使用,这里举例通过GCD创建主线程。
dispatch_async(dispatch_get_main_queue(), ^{
// 界面刷新操作...
});

你可能感兴趣的:(NSURLSession)