OCiOS开发:NSURLSession 网络请求

前言

iOS9 Https 网络协议

简介

  • iOS9把所有的http请求都改为https了:iOS9系统发送的网络请求将统一使用TLS 1.2 SSL。采用TLS 1.2 协议,目的是强制增强数据访问安全,而且系统 Foundation 框架下的相关网络请求,将不再默认使用 Http 等不安全的网络协议,而默认采用 TLS 1.2。服务器因此需要更新,以解析相关数据。如不更新,可通过在 Info.plist 中声明,倒退回不安全的网络请求。

ATS 简介

  • ATS全称为App Transport Security,它是iOS9的一个新特性,旨在提高iOS设备与服务器交互的安全性。简单地说,ATS会阻止未注册的网络请求。你可以在info.plist文件中注册相应的host,这样该host的网络请求就不会被阻止。你也可以设置不阻止任何host的网络请求。

解决iOS9以后Http网络协议接口奔溃方法

方案一: 升级服务器端,使用 TLS 1.2。

方案二:倒退回不安全的网络请求,即iOS9版本下不阻止任何网络访问。

方案二详细解决方法:

方法1:

  • 在info.plist文件中添加NSAppTransportSecurity字段,类型为Dictionary。

  • 在NSAppTransportSecurity字段下添加NSAllowsArbitraryLoads字段,类型为Boolean,值为YES。

方法2:

  • 选中Info.plist文件->鼠标右键选择Open AS->Source Code。

  • 在根字典下添加如下代码。

<key>NSAppTransportSecuritykey>
<dict>
    <key>NSAllowsArbitraryLoadskey>
    <true/>
dict>

同步、异步请求

  • 同步请求会在当前线程中执行网络请求操作,一般情况是在主线程执行。主线程默认处理用户交互信息与界面刷新,若执行网络请求,则在请求完成之前用户无法与界面交互。

  • 异步请求会新开辟一个线程,并在后台线程中执行请求操作。后台线程与主线程操作同时执行,不会影响到主线程处理用户界面交互信息。

GET、POST请求方法

  • GET和POST方法是HTTP请求中的常用方法。

  • GET请求常用于向服务器请求数据资源、POST请求常用于向服务器提交或同步数据。

  • GET请求的参数会跟在URL后进行传递,请求的数据会附在URL之后,参数可能会被第三方拦截,存在不安全因素。

  • POST请求参数是拼接在请求报文中的,并且可以进行相应的加密机制,相比GET请求有更好的安全性。

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)中分离出来。

NSURLSession 类套(suite of classes)

  • 苹果公司描述了新类NSURLSession,及其类套。包括新的上传,下载,处理认证等工具,能处理http协议中的所用事情。

  • 在编码前,重要的先理解它们是怎样协同工作的。

    OCiOS开发:NSURLSession 网络请求_第1张图片

  • NSurLSession由NSURLSessionConfiguration和可选代理(optional delegate)构成。再根据你的网络需求通过NSURLSessionTask来创建session。

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模式下运行。

常用方法

  • setHTTPAdditionalHeaders:设置头部/参数
// 设置所有的请求只接收JSON数据;
// 设置网络请求api所需的头部参数;
[configuration setHTTPAdditionalHeaders:@{@"Accept": @"application/json",
                                          @"apikey": @"159d95a8c050074fe3ca0df0d901fc62"}];

NSURLSession 网络会话

  • 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),以及上传和下载。其集成关系如下图:

OCiOS开发:NSURLSession 网络请求_第2张图片

OCiOS开发:NSURLSession 网络请求_第3张图片

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下载文件可以在下载中挂起,恢复。
创建方法
// 方法1
- (NSURLSessionDownloadTask *)downloadTaskWithRequest:(NSURLRequest *)request;  

// 方法2
- (NSURLSessionDownloadTask *)downloadTaskWithURL:(NSURL *)url;   

// 方法3
- (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]; 

Tips

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

数据请求成功之后的界面刷新操作

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

dispatch_async(dispatch_get_main_queue(), ^{
    // 界面刷新操作...
});

Get & Post Request Test

  • 故事板(storyboard)搭建界面

    OCiOS开发:NSURLSession 网络请求_第4张图片

  • 代码示例

#import "ViewController.h"

// get  net:http://apistore.baidu.com/apiworks/servicedetail/112.html
// post net:http://www.juhe.cn/docs/api/id/46/aid/131

#define NSLOG(OBJECT) NSLog(@"%@", OBJECT)

#define Get_Url_Apikey @"159d95a8c050074fe3ca0df0d901fc62"
#define Get_Url_String @"http://apis.baidu.com/apistore/weatherservice/weather?citypinyin=chengdu"

#define Post_Url_Apikey @"由于有次数限制,故请进入上方 post net 网址,注册聚合账号,申请数据,填入自己申请的appkey"
#define Post_Url_String @"http://apis.juhe.cn/cook/queryid"
#define Post_Url_Parameter_Id @(1001)
#define Post_Url_Parameter_Dtype @"json"

@interface ViewController ()

@end

@implementation ViewController

- (void)viewDidLoad {
    [super viewDidLoad];
}

#pragma mark - Request methods
- (IBAction)getRequest:(id)sender {

    // 1、构造URL资源地址
    NSURL *url = [NSURL URLWithString:Get_Url_String];
    // 2、创建Request请求
    NSMutableURLRequest *request = [NSMutableURLRequest requestWithURL:url];
    // 3、配置Request请求
    // 设置请求方法
    request.HTTPMethod = @"GET";
    // 设置请求超时
    request.timeoutInterval = 10.0;
    // 设置头部参数
    [request addValue:Get_Url_Apikey forHTTPHeaderField:@"apikey"];

    // 4、构造NSURLSessionConfiguration
    NSURLSessionConfiguration *configuration = [NSURLSessionConfiguration defaultSessionConfiguration];
    // 5、构造NSURLSession,网络会话;
    NSURLSession *session = [NSURLSession sessionWithConfiguration:configuration];;
    // 6、构造NSURLSessionTask,会话任务;
    NSURLSessionTask *task = [session dataTaskWithRequest:request completionHandler:^(NSData * _Nullable data, NSURLResponse * _Nullable response, NSError * _Nullable error) {

        // 请求失败,打印错误信息
        if (error) {
            NSLOG(error.localizedDescription);
        }
        // 8、请求成功,解析数据
        else {
            // 直接将data数据转成OC字符串(NSUTF8StringEncoding);
            // NSString *dataString = [[NSString alloc] initWithData:data encoding:NSUTF8StringEncoding];

            // JSON数据格式解析
           id object = [NSJSONSerialization JSONObjectWithData:data options:NSJSONReadingMutableLeaves error:&error];
            // 9、判断是否解析成功
            if (error) {
                NSLOG(error.localizedDescription);
            }else {
                NSLOG(object);
                // 解析成功,处理数据,通过GCD获取主队列,在主线程中刷新界面。
                dispatch_async(dispatch_get_main_queue(), ^{
                    // 刷新界面....
                });
            }
        }
    }];
    // 7、执行任务
    [task resume];
}
- (IBAction)postRequest:(id)sender {

    // 1、创建URL资源地址
    NSURL *url = [NSURL URLWithString:Post_Url_String];
    // 2、创建Reuest请求
    NSMutableURLRequest *request = [NSMutableURLRequest requestWithURL:url];
    // 3、配置Request
    request.timeoutInterval = 10.0;
    request.HTTPMethod = @"POST";
    // 4、构造请求参数
    // 4.1、创建字典参数,将参数放入字典中,可防止程序员在主观意识上犯错误,即参数写错。
    NSDictionary *parametersDict = @{@"id":Post_Url_Parameter_Id, @"key":Post_Url_Apikey, @"dtype":Post_Url_Parameter_Dtype};
    // 4.2、遍历字典,以“key=value&”的方式创建参数字符串。
    NSMutableString *parameterString = [NSMutableString string];

    for (NSString *key in parametersDict.allKeys) {
        // 拼接字符串
        [parameterString appendFormat:@"%@=%@&", key, parametersDict[key]];
    }
    // 4.3、截取参数字符串,去掉最后一个“&”,并且将其转成NSData数据类型。
    NSData *parametersData = [[parameterString substringToIndex:parameterString.length - 1] dataUsingEncoding:NSUTF8StringEncoding];

    // 5、设置请求报文
    request.HTTPBody = parametersData;
    // 6、构造NSURLSessionConfiguration
    NSURLSessionConfiguration *configuration = [NSURLSessionConfiguration defaultSessionConfiguration];
    // 7、创建网络会话
    NSURLSession *session = [NSURLSession sessionWithConfiguration:configuration];
    // 8、创建会话任务
    NSURLSessionTask *task = [session dataTaskWithRequest:request completionHandler:^(NSData * _Nullable data, NSURLResponse * _Nullable response, NSError * _Nullable error) {
        // 10、判断是否请求成功
        if (error) {
            NSLOG(error.localizedDescription);
        }else {
            // 如果请求成功,则解析数据。
            id object = [NSJSONSerialization JSONObjectWithData:data options:NSJSONReadingMutableLeaves error:&error];
            // 11、判断是否解析成功
            if (error) {
                NSLOG(error.localizedDescription);
            }else {
                // 解析成功,处理数据,通过GCD获取主队列,在主线程中刷新界面。
                NSLOG(object);
                dispatch_async(dispatch_get_main_queue(), ^{
                    // 刷新界面...
                });
            }
        }

    }];
    // 9、执行任务
    [task resume];

}

@end

你可能感兴趣的:(Objective-c,应用开发)