方案一: 升级服务器端,使用 TLS 1.2。
方案二:倒退回不安全的网络请求,即iOS9版本下不阻止任何网络访问。
方案二详细解决方法:
方法1:
在info.plist文件中添加NSAppTransportSecurity字段,类型为Dictionary。
在NSAppTransportSecurity字段下添加NSAllowsArbitraryLoads字段,类型为Boolean,值为YES。
方法2:
选中Info.plist文件->鼠标右键选择Open AS->Source Code。
在根字典下添加如下代码。
<key>NSAppTransportSecurity</key>
<dict>
<key>NSAllowsArbitraryLoads</key>
<true/>
</dict>
同步请求会在当前线程中执行网络请求操作,一般情况是在主线程执行。主线程默认处理用户交互信息与界面刷新,若执行网络请求,则在请求完成之前用户无法与界面交互。
异步请求会新开辟一个线程,并在后台线程中执行请求操作。后台线程与主线程操作同时执行,不会影响到主线程处理用户界面交互信息。
GET和POST方法是HTTP请求中的常用方法。
GET请求常用于向服务器请求数据资源、POST请求常用于向服务器提交或同步数据。
GET请求的参数会跟在URL后进行传递,请求的数据会附在URL之后,参数可能会被第三方拦截,存在不安全因素。
POST请求参数是拼接在请求报文中的,并且可以进行相应的加密机制,相比GET请求有更好的安全性。
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的时候配置一个选项,就能得到后台网络的所有好处。这样可以延长电池寿命,并且还支持UIKit的多task,在进程间使用相同的委托模型。
能够暂停和恢复网络操作:使用NSURLSession API能够暂停,停止,恢复所有的网络任务,再也完全不需要子类化NSOperation。
可配置的容器:对于NSURLSession里面的requests来说,每个NSURLSession都是可配置的容器。举个例来说,假如你需要设置HTTP header选项,你只用做一次,session里面的每个request就会有同样的配置。
提高认证处理:认证是在一个指定的连接基础上完成的。在使用NSURLConnection时,如果发出一个访问,会返回一个任意的request。此时,你就不能确切的知道哪个request收到了访问。而在NSURLSession中,就能用代理处理认证。
丰富的代理模式:在处理认证的时候,NSURLConnection有一些基于异步的block方法,但是它的代理方法就不能处理认证,不管请求是成功或是失败。在NSURLSession中,可以混合使用代理和block方法处理认证。
上传和下载通过文件系统:它鼓励将数据(文件内容)从元数据(URL和settings)中分离出来。
苹果公司描述了新类NSURLSession,及其类套。包括新的上传,下载,处理认证等工具,能处理http协议中的所用事情。
在编码前,重要的先理解它们是怎样协同工作的。
NSurLSession由NSURLSessionConfiguration和可选代理(optional delegate)构成。再根据你的网络需求通过NSURLSessionTask来创建session。
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模式下运行。
// 设置所有的请求只接收JSON数据;
// 设置网络请求api所需的头部参数;
[configuration setHTTPAdditionalHeaders:@{@"Accept": @"application/json",
@"apikey": @"159d95a8c050074fe3ca0df0d901fc62"}];
创建一个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收到这个事件之后会被解引用。
// 1、通过request创建
- (NSURLSessionDataTask *)dataTaskWithRequest:(NSURLRequest *)request; // 2、通过url创建 - (NSURLSessionDataTask *)dataTaskWithURL:(NSURL *)url;
// 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;
// 1、通过指定文件源上传
- (NSURLSessionUploadTask *)uploadTaskWithRequest:(NSURLRequest *)request fromFile:(NSURL *)fileURL; // 2、通过指定数据源上传 - (NSURLSessionUploadTask *)uploadTaskWithRequest:(NSURLRequest *)request fromData:(NSData *)bodyData; // 3、通过request上传 - (NSURLSessionUploadTask *)uploadTaskWithStreamedRequest:(NSURLRequest *)request;
- (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;
// 方法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、下载失败或是取消还可以得到已经下载的那一部分数据。
[task resume];
Tips
同时管理多个task时,可以用taskIdentifier属性来唯一标识task。
在iOS中,默认只有一个线程,称为主线程,主线程主要处理用户交互和界面刷新,通过NSURLSession请求数据,一般会开辟子线程进行,以防止主线程拥堵,导致界面卡死甚至奔溃。因此在获取到数据之后,刷新界面应该在主线程执行刷新操作。获取主线程刷新界面的方法有多重方式,多线程章节会详细讲解线程创建使用,这里举例通过GCD创建主线程。
dispatch_async(dispatch_get_main_queue(), ^{
// 界面刷新操作...
});
#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