互联网伏乞方法总括复习

ca88会员登录中心 3
ca88会员登录中心
2.0版本:

使用 AFHTTPRequestOperationManager这个网络管理者

AFHTTPRequestOperationManager *manager = [AFHTTPRequestOperationManager manager];[manager GET:@"需要请求的url" parameters:nil success:^(AFHTTPRequestOperation *operation, id responseObject) { NSLog;} failure:^(AFHTTPRequestOperation *operation, NSError*error) { NSLog;}];

介绍下2.0版本跟3.0版本使用的区别

使用 AFHTTPRequestOperationManager这个网络管理者

AFHTTPRequestOperationManager *manager =
[AFHTTPRequestOperationManager manager];

[manager GET:@”需要请求的url” parameters:nil
success:^(AFHTTPRequestOperation *operation, id responseObject) {

NSLog(@”请求成功”);

} failure:^(AFHTTPRequestOperation *operation, NSError*error) {

NSLog(@”请求失败”);

}];

3.0版本:

使用 AFHTTPSessionManager 这个网络管理者

AFHTTPSessionManager *session = [AFHTTPSessionManager manager];

[session GET:@”需要请求的url” parameters:nil
success:^(NSURLSessionDataTask *task, id responseObject) {

NSLog(@”请求成功”);

} failure:^(NSURLSessionDataTask *task, NSError *error) {

NSLog(@”请求失败”);

}];

  • ② 下载完的事件采用delegate形式的API
3.0版本:

使用 AFHTTPSessionManager 这个网络管理者

AFHTTPSessionManager *session = [AFHTTPSessionManager manager];[session GET:@"需要请求的url" parameters:nil success:^(NSURLSessionDataTask *task, id responseObject) { NSLog;} failure:^(NSURLSessionDataTask *task, NSError *error) { NSLog; }];

AFNetWorking的使用

GET请求

-(void)get

{

//1.创建会话管理者

AFHTTPSessionManager *manager = [AFHTTPSessionManager manager];

//2.封装参数

NSDictionary *dict = @{

@”username”:@”Lion”,

@”pwd”:@”1314″,

@”type”:@”JSON”

};

//3.发送GET请求

/*

第一个参数:请求路径(NSString)+ 不需要加参数

第二个参数:发送给服务器的参数数据

第三个参数:progress 进度回调

第四个参数:success  成功之后的回调(此处的成功或者是失败指的是整个请求)

task:请求任务

responseObject:注意!!!响应体信息—>(json—>oc))

task.response: 响应头信息

第五个参数:failure 失败之后的回调

*/

[manager GET:@”需要请求的URL” parameters:dict progress:nil
success:^(NSURLSessionDataTask * _Nonnull task, id  _Nullable
responseObject) {

NSLog(@”success–%@–%@”,[responseObject class],responseObject);

} failure:^(NSURLSessionDataTask * _Nullable task, NSError *
_Nonnull error) {

NSLog(@”failure–%@”,error);

}];

}

POST请求和GET请求一样,只需要换成POST请求方法

[manager POST:@”需要请求的URL” parameters:dict progress:nil
success:^(NSURLSessionDataTask * _Nonnull task, id  _Nullable
responseObject) {

NSLog(@”success–%@–%@”,[responseObject class],responseObject);

} failure:^(NSURLSessionDataTask * _Nullable task, NSError *
_Nonnull error) {

NSLog(@”failure–%@”,error);

}];

下载操作(不是离线下载,离线下载需要自己手动处理)

-(void)download

{

//1.创建会话管理者

AFHTTPSessionManager *manager = [AFHTTPSessionManager manager];

//2.确定请求路径

ca88会员登录中心,NSURL *url = [NSURL URLWithString:@”需要请求的URL”];

//3.创建请求对象

NSURLRequest *request = [NSURLRequest requestWithURL:url];

//4.发送网络请求下载文件

/*

第一个参数:请求对象

第二个参数:progress 进度回调

downloadProgress

@property int64_t totalUnitCount;

@property int64_t completedUnitCount;

第三个参数:destination 让我们告诉系统应该把文件存放到什么地方

内部自动的完成剪切处理

第四个参数: completionHandler 完成之后的回调

response 响应头信息

filePath  文件最终的存储路径

error 错误信息

*/

[[manager downloadTaskWithRequest:request progress:^(NSProgress *
_Nonnull downloadProgress) {

NSLog(@”%f”,1.0 * downloadProgress.completedUnitCount /
downloadProgress.totalUnitCount);

} destination:^NSURL * _Nonnull(NSURL * _Nonnull targetPath,
NSURLResponse * _Nonnull response) {

//拼接文件的全路径

NSString *fullpath =
[[NSSearchPathForDirectoriesInDomains(NSCachesDirectory,
NSUserDomainMask, YES) lastObject]
stringByAppendingPathComponent:response.suggestedFilename];

NSLog(@”fullpath == %@”,fullpath);

return [NSURL fileURLWithPath:fullpath];

} completionHandler:^(NSURLResponse * _Nonnull response, NSURL *
_Nullable filePath, NSError * _Nullable error) {

NSLog(@”%@”,filePath);

}] resume];

}

上传操作

-(void)upload

{

//1.创建会话管理者

AFHTTPSessionManager *manager = [AFHTTPSessionManager manager];

//2.发送请求上传文件

/*

第一个参数:请求路径(NSString)

第二个参数:非文件参数

第三个参数:constructingBodyWithBlock
拼接数据(告诉AFN要上传的数据是哪些)

第四个参数:progress 进度回调

第五个参数:success 成功回调

responseObject:响应体

第六个参数:failure 失败的回调

*/

[manager POST:@”需要请求的URL” parameters:nil
constructingBodyWithBlock:^(id  _Nonnull formData) {

NSData *data = [NSData
dataWithContentsOfFile:@”/Users/apple/Desktop/Snip20160409_148.png”];

//拼接数据

/*

第一个参数:文件参数 (二进制数据)

第二个参数:参数名~file

第三个参数:该文件上传到服务器以什么名称来保存

第四个参数:

*/

[formData appendPartWithFileData:data name:@”file”
fileName:@”123.png” mimeType:@”image/png”];

} progress:^(NSProgress * _Nonnull uploadProgress) {

NSLog(@”%f”,1.0 * uploadProgress.completedUnitCount /
uploadProgress.totalUnitCount);

} success:^(NSURLSessionDataTask * _Nonnull task, id  _Nullable
responseObject) {

NSLog(@”success–%@”,responseObject);

} failure:^(NSURLSessionDataTask * _Nullable task, NSError *
_Nonnull error) {

NSLog(@”failure — %@”,error);

}];

}

使用NSURLSession发送GET请求的方法和NSURLConnection类似,整个过程如下:

AFNetWorking的使用

  • GET请求

-get{ //1.创建会话管理者 AFHTTPSessionManager *manager = [AFHTTPSessionManager manager]; //2.封装参数 NSDictionary *dict = @{ @"username":@"Lion", @"pwd":@"1314", @"type":@"JSON" }; //3.发送GET请求 /* 第一个参数:请求路径+ 不需要加参数 第二个参数:发送给服务器的参数数据 第三个参数:progress 进度回调 第四个参数:success 成功之后的回调(此处的成功或者是失败指的是整个请求) task:请求任务 responseObject:注意!!!响应体信息--->(json--->oc)) task.response: 响应头信息 第五个参数:failure 失败之后的回调 */ [manager GET:@"需要请求的URL" parameters:dict progress:nil success:^(NSURLSessionDataTask * _Nonnull task, id _Nullable responseObject) { NSLog(@"success--%@--%@",[responseObject class],responseObject); } failure:^(NSURLSessionDataTask * _Nullable task, NSError * _Nonnull error) { NSLog(@"failure--%@",error); }];}
  • POST请求和GET请求一样,只需要换成POST请求方法

 [manager POST:@"需要请求的URL" parameters:dict progress:nil success:^(NSURLSessionDataTask * _Nonnull task, id _Nullable responseObject) { NSLog(@"success--%@--%@",[responseObject class],responseObject); } failure:^(NSURLSessionDataTask * _Nullable task, NSError * _Nonnull error) { NSLog(@"failure--%@",error); }];
  • 下载操作(不是离线下载,离线下载需要自己手动处理)

-download{ //1.创建会话管理者 AFHTTPSessionManager *manager = [AFHTTPSessionManager manager]; //2.确定请求路径 NSURL *url = [NSURL URLWithString:@"需要请求的URL"]; //3.创建请求对象 NSURLRequest *request = [NSURLRequest requestWithURL:url]; //4.发送网络请求下载文件 /* 第一个参数:请求对象 第二个参数:progress 进度回调 downloadProgress @property int64_t totalUnitCount; @property int64_t completedUnitCount; 第三个参数:destination 让我们告诉系统应该把文件存放到什么地方 内部自动的完成剪切处理 第四个参数: completionHandler 完成之后的回调 response 响应头信息 filePath 文件最终的存储路径 error 错误信息 */ [[manager downloadTaskWithRequest:request progress:^(NSProgress * _Nonnull downloadProgress) { NSLog(@"%f",1.0 * downloadProgress.completedUnitCount / downloadProgress.totalUnitCount); } destination:^NSURL * _Nonnull(NSURL * _Nonnull targetPath, NSURLResponse * _Nonnull response) { //拼接文件的全路径 NSString *fullpath = [[NSSearchPathForDirectoriesInDomains(NSCachesDirectory, NSUserDomainMask, YES) lastObject] stringByAppendingPathComponent:response.suggestedFilename]; NSLog(@"fullpath == %@",fullpath); return [NSURL fileURLWithPath:fullpath]; } completionHandler:^(NSURLResponse * _Nonnull response, NSURL * _Nullable filePath, NSError * _Nullable error) { NSLog(@"%@",filePath); }] resume]; }
  • 上传操作

-upload{ //1.创建会话管理者 AFHTTPSessionManager *manager = [AFHTTPSessionManager manager]; //2.发送请求上传文件 /* 第一个参数:请求路径 第二个参数:非文件参数 第三个参数:constructingBodyWithBlock 拼接数据(告诉AFN要上传的数据是哪些) 第四个参数:progress 进度回调 第五个参数:success 成功回调 responseObject:响应体 第六个参数:failure 失败的回调 */ [manager POST:@"需要请求的URL" parameters:nil constructingBodyWithBlock:^(id<AFMultipartFormData> _Nonnull formData) { NSData *data = [NSData dataWithContentsOfFile:@"/Users/apple/Desktop/Snip20160409_148.png"]; //拼接数据 /* 第一个参数:文件参数  第二个参数:参数名~file 第三个参数:该文件上传到服务器以什么名称来保存 第四个参数: */ [formData appendPartWithFileData:data name:@"file" fileName:@"123.png" mimeType:@"image/png"]; } progress:^(NSProgress * _Nonnull uploadProgress) { NSLog(@"%f",1.0 * uploadProgress.completedUnitCount / uploadProgress.totalUnitCount); } success:^(NSURLSessionDataTask * _Nonnull task, id _Nullable responseObject) { NSLog(@"success--%@",responseObject); } failure:^(NSURLSessionDataTask * _Nullable task, NSError * _Nonnull error) { NSLog(@"failure -- %@",error); }];}

NSURLSessionConfiguration配置信息

作用:

统一配置session的信息,每个session可以发送多个请求

设置蜂窝网络,隐私信息,超时时间\

@property (strong, nonatomic) NSURLSession *session;

//将session设置为全局的属性,方便发送网络请求

//在懒加载里面设置配置的信息,达到统一设置

#pragma  makr – 懒加载

-(NSURLSession *)session

{

if (_session == nil) {

//设置配置信息

NSURLSessionConfiguration *config = [NSURLSessionConfiguration
defaultSessionConfiguration];

//统一设置请求超时

config.timeoutIntervalForRequest = 15.0;

//设置是否允许蜂窝网络访问

config.allowsCellularAccess = YES;

_session = [NSURLSession sessionWithConfiguration:config
delegate:self delegateQueue:[NSOperationQueue mainQueue]];

}

return _session;

}

- downloadBtnClicked:(UIButton *)sender { // 创建下载路径 NSURL *url = [NSURL URLWithString:@"http://dldir1.qq.com/qqfile/QQforMac/QQ_V5.4.0.dmg"]; // NSURLConnection发送异步Get请求,并实现相应的代理方法,该方法iOS9.0之后废除了。 [NSURLConnection connectionWithRequest:[NSURLRequest requestWithURL:url] delegate:self];}

介绍下2.0版本跟3.0版本使用的区别

为什么会废弃NSURLConnection而使用NSURLSession这个网络类呢?

从iOS7之后,苹果就推出了NSURLSession ,
AFN跟随着苹果的脚步,加入了对其的支持

NSURLSession推出后苹果大力推广,但是其效果和NSURLConnection效果差不多,为什么NSURLConnection还会被废弃呢?

因为:2015年的WWDC大会,HTTP2.0时代的到来,(1.1版本是99年),HTTP2.0比之前的版本速度更快

iOS9(Xcode7)之后,
NSURLSession开始正式支持HTTP2.0,所以相比速度快差不多4倍,
NSURLSession将NSURLConnection甩开了距离

NSURLConnection作为过气的框架,作为对比了解一下还是有必要的。NSURLSession作为众多网络相关的第三方框架基于的苹果原生框架,更是有必要学习总结一下。作为第三方框架,AFNetworking,SDWebImage等等其实它们的老版本是基于NSURLConnection封装而成的,后来才改成的基于NSURLSession。这些第三方框架相比原生框架封装了缓存的逻辑,比如内存缓存,磁盘缓存,操作缓存等等。

AFNetWorking的介绍(给没使用过的读者看的)

  • AFNetWorking是目前iOS开发者网络库中最多的选择
  • AFNetWorking是对NSURLConnection和NSURLSession的封装,iOS
    9之后删除了NSURLConnection的API的所有支持,完全基于NSURLSession
    的API.

AFNetWorking的介绍

分享一个很详细的总结

www.cocoachina.com/ios/20161209/18277.html

ca88会员登录中心 1

AFNetWorking是目前iOS开发者网络库中最多的选择

AFNetWorking是对NSURLConnection和NSURLSession的封装,iOS
9(AFN3.0版本)之后删除了NSURLConnection的API的所有支持,完全基于NSURLSession
的API.

  • 调用示例 — 采用delegate的API ②

AFNetWorking的序列化

  • AFNetWorking默认是JSON解析(因为开发中百分之90都是JSON数据)

默认情况:JSON
AFJSONResponseSerializerXML:AFXMLParserResponseSerializer既不是XML也不是JSON:AFHTTPResponseSerializer

  • JSON

 //1.创建会话管理者 AFHTTPSessionManager *manager = [AFHTTPSessionManager manager]; /* 1)afn内部默认已经完成了JSON解析工作 优点:方便 缺点:如果服务器返回的数据不是JSON会报错 */ NSDictionary *dict = @{@"type":@"JSON"}; //2.发请求 [manager GET:@"需要请求的URL" parameters:dict progress:nil success:^(NSURLSessionDataTask * _Nonnull task, id _Nullable responseObject) { NSLog(@"%@--%@",[responseObject class],responseObject); } failure:^(NSURLSessionDataTask * _Nullable task, NSError * _Nonnull error) { NSLog(@"%@---",error); }];
  • XML

-xml{<NSXMLParserDelegate>代理协议 //1.创建会话管理者 AFHTTPSessionManager *manager = [AFHTTPSessionManager manager]; //设置以XML的方式来解析数据 manager.responseSerializer = [AFXMLParserResponseSerializer serializer]; NSDictionary *dict = @{@"type":@"XML"}; //2.发请求 [manager GET:@"需要请求的URL" parameters:dict progress:nil success:^(NSURLSessionDataTask * _Nonnull task, id _Nullable responseObject) { NSLog(@"%@--%@",[responseObject class],responseObject); //1.创建解析器 NSXMLParser *parser = (NSXMLParser *)responseObject; //2.设置代理 parser.delegate = self; //3.开始解析 [parser parse]; } failure:^(NSURLSessionDataTask * _Nullable task, NSError * _Nonnull error) { NSLog(@"%@---",error); }];}
  • OtherHttpData

-otherHttpData{ //1.创建会话管理者 AFHTTPSessionManager *manager = [AFHTTPSessionManager manager]; //设置不做处理 manager.responseSerializer = [AFHTTPResponseSerializer serializer]; //2.发请求 [manager GET:@"需要请求的URL" parameters:nil progress:nil success:^(NSURLSessionDataTask * _Nonnull task, id _Nullable responseObject) { NSLog(@"%@--%@",[responseObject class],responseObject); } failure:^(NSURLSessionDataTask * _Nullable task, NSError * _Nonnull error) { NSLog(@"%@---",error); }];}#pragma mark NSXMLParserDelegate-parser:(NSXMLParser *)parser didStartElement:(NSString *)elementName namespaceURI:(NSString *)namespaceURI qualifiedName:(NSString *)qName attributes:(NSDictionary<NSString *,NSString *> *)attributeDict{ NSLog(@"%@--%@",elementName,attributeDict);}
  • 检测网络的状态

-networkStatusChangeAFN{ //1.获得一个网络状态监听管理者 AFNetworkReachabilityManager *manager = [AFNetworkReachabilityManager sharedManager]; //2.监听状态的改变(当网络状态改变的时候就会调用该block) [manager setReachabilityStatusChangeBlock:^(AFNetworkReachabilityStatus status) { /* AFNetworkReachabilityStatusUnknown = -1, 未知 AFNetworkReachabilityStatusNotReachable = 0, 没有网络 AFNetworkReachabilityStatusReachableViaWWAN = 1, 3G|4G AFNetworkReachabilityStatusReachableViaWiFi = 2, WIFI */ switch  { case AFNetworkReachabilityStatusReachableViaWiFi: NSLog; break; case AFNetworkReachabilityStatusReachableViaWWAN: NSLog; break; case AFNetworkReachabilityStatusNotReachable: NSLog; break; case AFNetworkReachabilityStatusUnknown: NSLog; break; default: break; } }]; //3.手动开启 开始监听 [manager startMonitoring];}

AFN使用技巧1.在开发的时候可以创建一个工具类,继承自我们的AFN中的请求管理者,再控制器中真正发请求的代码使用自己封装的工具类。2.这样做的优点是以后如果修改了底层依赖的框架,那么我们修改这个工具类就可以了,而不用再一个一个的去修改。3.该工具类一般提供一个单例方法,在该方法中会设置一个基本的请求路径。4.该方法通常还会提供对GET或POST请求的封装。5.在外面的时候通过该工具类来发送请求6.单例方法:+ (instancetype)shareNetworkTools{ static XMGNetworkTools *instance; static dispatch_once_t onceToken; dispatch_once(&onceToken, ^{ // 注意: BaseURL中一定要以/结尾 instance=[[selfalloc]initWithBaseURL:[NSURL URLWithString:@"请求的URL"]]; }); return instance;} 

AFNetworking流程图

ca88会员登录中心 2

为什么会废弃NSURLConnection而使用NSURLSession这个网络类呢?

  • 从iOS7之后,苹果就推出了NSURLSession ,
    AFN跟随着苹果的脚步,加入了对其的支持
  • NSURLSession推出后苹果大力推广,但是其效果和NSURLConnection效果差不多,为什么NSURLConnection还会被废弃呢?
  • 因为:2015年的WWDC大会,HTTP2.0时代的到来,(1.1版本是99年),HTTP2.0比之前的版本速度更快
  • iOS9之后, NSURLSession开始正式支持HTTP2.0,所以相比速度快差不多4倍,
    NSURLSession将NSURLConnection甩开了距离
  • 纯属个人理解

AFNetWorking的序列化

AFNetWorking默认是JSON解析(因为开发中百分之90都是JSON数据)

默认情况:JSON  AFJSONResponseSerializer

XML:AFXMLParserResponseSerializer

既不是XML也不是JSON:AFHTTPResponseSerializer

JSON

//1.创建会话管理者

AFHTTPSessionManager *manager = [AFHTTPSessionManager manager];

/*

1)afn内部默认已经完成了JSON解析工作

优点:方便

缺点:如果服务器返回的数据不是JSON会报错

*/

NSDictionary *dict = @{@”type”:@”JSON”};

//2.发请求

[manager GET:@”需要请求的URL” parameters:dict progress:nil
success:^(NSURLSessionDataTask * _Nonnull task, id  _Nullable
responseObject) {

NSLog(@”%@–%@”,[responseObject class],responseObject);

} failure:^(NSURLSessionDataTask * _Nullable task, NSError *
_Nonnull error) {

NSLog(@”%@—“,error);

}];

XML

-(void)xml

{

代理协议

//1.创建会话管理者

AFHTTPSessionManager *manager = [AFHTTPSessionManager manager];

//设置以XML的方式来解析数据

manager.responseSerializer = [AFXMLParserResponseSerializer
serializer];

NSDictionary *dict = @{@”type”:@”XML”};

//2.发请求

[manager GET:@”需要请求的URL” parameters:dict progress:nil
success:^(NSURLSessionDataTask * _Nonnull task, id  _Nullable
responseObject) {

NSLog(@”%@–%@”,[responseObject class],responseObject);

//1.创建解析器

NSXMLParser *parser = (NSXMLParser *)responseObject;

//2.设置代理

parser.delegate = self;

//3.开始解析

[parser parse];

} failure:^(NSURLSessionDataTask * _Nullable task, NSError *
_Nonnull error) {

NSLog(@”%@—“,error);

}];

}

OtherHttpData

-(void)otherHttpData

{

//1.创建会话管理者

AFHTTPSessionManager *manager = [AFHTTPSessionManager manager];

//设置不做处理

manager.responseSerializer = [AFHTTPResponseSerializer serializer];

//2.发请求

[manager GET:@”需要请求的URL” parameters:nil progress:nil
success:^(NSURLSessionDataTask * _Nonnull task, id  _Nullable
responseObject) {

NSLog(@”%@–%@”,[responseObject class],responseObject);

} failure:^(NSURLSessionDataTask * _Nullable task, NSError *
_Nonnull error) {

NSLog(@”%@—“,error);

}];

}

#pragma mark NSXMLParserDelegate

-(void)parser:(NSXMLParser *)parser didStartElement:(NSString
*)elementName namespaceURI:(NSString *)namespaceURI
qualifiedName:(NSString *)qName attributes:(NSDictionary
*)attributeDict

{

NSLog(@”%@–%@”,elementName,attributeDict);

}

检测网络的状态

-(void)networkStatusChangeAFN

{

//1.获得一个网络状态监听管理者

AFNetworkReachabilityManager *manager = 
[AFNetworkReachabilityManager sharedManager];

//2.监听状态的改变(当网络状态改变的时候就会调用该block)

[manager
setReachabilityStatusChangeBlock:^(AFNetworkReachabilityStatus status)
{

/*

AFNetworkReachabilityStatusUnknown          = -1,  未知

AFNetworkReachabilityStatusNotReachable    = 0,  没有网络

AFNetworkReachabilityStatusReachableViaWWAN = 1,    3G|4G

AFNetworkReachabilityStatusReachableViaWiFi = 2,  WIFI

*/

switch (status) {

case AFNetworkReachabilityStatusReachableViaWiFi:

NSLog(@”wifi”);

break;

case AFNetworkReachabilityStatusReachableViaWWAN:

NSLog(@”3G|4G”);

break;

case AFNetworkReachabilityStatusNotReachable:

NSLog(@”没有网络”);

break;

case AFNetworkReachabilityStatusUnknown:

NSLog(@”未知”);

break;

default:

break;

}

}];

3.手动开启 开始监听

[manager startMonitoring];

}

AFN使用技巧

1.在开发的时候可以创建一个工具类,继承自我们的AFN中的请求管理者,再控制器中真正发请求的代码使用自己封装的工具类。

2.这样做的优点是以后如果修改了底层依赖的框架,那么我们修改这个工具类就可以了,而不用再一个一个的去修改。

3.该工具类一般提供一个单例方法,在该方法中会设置一个基本的请求路径。

4.该方法通常还会提供对GET或POST请求的封装。

5.在外面的时候通过该工具类来发送请求

6.单例方法:+ (instancetype)shareNetworkTools{

static XMGNetworkTools *instance;

static dispatch_once_t onceToken;

dispatch_once(&onceToken, ^{

// 注意: BaseURL中一定要以/结尾

instance=[[selfalloc]initWithBaseURL:[NSURL
URLWithString:@”请求的URL”]];    });    return instance;

}

NSURLSessionConfiguration配置信息

  • 作用:
  • 统一配置session的信息,每个session可以发送多个请求
  • 设置蜂窝网络,隐私信息,超时时间\

@property (strong, nonatomic) NSURLSession *session;//将session设置为全局的属性,方便发送网络请求//在懒加载里面设置配置的信息,达到统一设置#pragma makr - 懒加载-(NSURLSession *)session{ if (_session == nil) { //设置配置信息 NSURLSessionConfiguration *config = [NSURLSessionConfiguration defaultSessionConfiguration]; //统一设置请求超时 config.timeoutIntervalForRequest = 15.0; //设置是否允许蜂窝网络访问 config.allowsCellularAccess = YES; _session = [NSURLSession sessionWithConfiguration:config delegate:self delegateQueue:[NSOperationQueue mainQueue]]; } return _session;}

在iOS9.0之后,以前使用的NSURLConnection过期,苹果推荐使用NSURLSession来替换NSURLConnection完成网路请求相关操作。NSURLSession的使用非常简单,先根据会话对象创建一个请求Task,然后执行该Task即可。NSURLSessionTask本身是一个抽象类,在使用的时候,通常是根据具体的需求使用它的几个子类。关系如下:

① 下载完的事件采用block形式
  • get请求示例1
  • 关键API
  • sharedSession
  • requestWithURL:
  • dataTaskWithRequest:
-getWithBlock1{ //对请求路径的说明 //http://120.25.226.186:32812/login?username=520it&pwd=520&type=JSON //协议头+主机地址+接口名称+?+参数1&参数2&参数3 //协议头+主机地址(120.25.226.186:32812)+接口名称+?+参数1(username=520it)&参数2&参数3(type=JSON) //GET请求,直接把请求参数跟在URL的后面以?隔开,多个参数之间以&符号拼接 //1.确定请求路径 NSURL *url = [NSURL URLWithString:@"http://120.25.226.186:32812/login?username=520it&pwd=520it&type=JSON"]; //2.创建请求对象 //请求对象内部默认已经包含了请求头和请求方法 NSURLRequest *request = [NSURLRequest requestWithURL:url]; //3.获得会话对象 NSURLSession *session = [NSURLSession sharedSession]; //4.根据会话对象创建一个Task /* 第一个参数:请求对象 第二个参数:completionHandler回调(请求完成的回调) data:响应体信息 response:响应头信息,主要是对服务器端的描述 error:错误信息,如果请求失败,则error有值 */ NSURLSessionDataTask *dataTask = [session dataTaskWithRequest:request completionHandler:^(NSData * _Nullable data, NSURLResponse * _Nullable response, NSError * _Nullable error) { if (error == nil) { //6.解析服务器返回的数据 //说明:(此处返回的数据是JSON格式的,因此使用NSJSONSerialization进行反序列化处理) NSDictionary *dict = [NSJSONSerialization JSONObjectWithData:data options:kNilOptions error:nil]; NSLog(@"%@",dict); } }]; //5.执行任务 [dataTask resume];}
  • get请求示例2
  • 关键API

sharedSessiondataTaskWithURL:

-getWithBlock2{ //1.确定请求路径 NSURL *url = [NSURL URLWithString:@"http://120.25.226.186:32812/login?username=520it&pwd=520it&type=JSON"]; //2.获得会话对象 NSURLSession *session = [NSURLSession sharedSession]; //3.根据会话对象创建一个Task /* 第一个参数:请求路径 第二个参数:completionHandler回调(请求完成的回调) data:响应体信息 response:响应头信息,主要是对服务器端的描述 error:错误信息,如果请求失败,则error有值 注意: 1)该方法内部会自动将请求路径包装成一个请求对象,该请求对象默认包含了请求头信息和请求方法如果要发送的是POST请求,则不能使用该方法 */ NSURLSessionDataTask *dataTask = [session dataTaskWithURL:url completionHandler:^(NSData * _Nullable data, NSURLResponse * _Nullable response, NSError * _Nullable error) { //5.解析数据 NSDictionary *dict = [NSJSONSerialization JSONObjectWithData:data options:kNilOptions error:nil]; NSLog(@"%@",dict); }]; //4.执行任务 [dataTask resume];}
/** * 点击按钮 -- 使用NSURLConnection下载图片文件,并显示再imageView上 */- downloadBtnClicked:(UIButton *)sender { // 创建下载路径 NSURL *url = [NSURL URLWithString:@"https://upload-images.jianshu.io/upload_images/1877784-b4777f945878a0b9.jpg"]; // NSURLConnection发送异步Get请求,该方法iOS9.0之后就废除了,推荐NSURLSession [NSURLConnection sendAsynchronousRequest:[NSURLRequest requestWithURL:url] queue:[NSOperationQueue mainQueue] completionHandler:^(NSURLResponse * _Nullable response, NSData * _Nullable data, NSError * _Nullable connectionError) { self.imageView.image = [UIImage imageWithData:data]; // 可以在这里把下载的文件保存 }];}

iOS的开发中的网络下载方式包括NSData(最原始,实际开发基本不会用),NSURLConnection(古老又过气的苹果原生网络框架),NSURLSession(现在流行的苹果网络框架),AFNetworking,SDWebImage以及基于AFNetworking的二次封装框架例如XMNetworking,HYBNetworking等等。

  • 下载示例
  • ① 下载完的事件采用block形式的API

1)确定请求路径(一般由公司的后台开发人员以接口文档的方式提供),GET请求参数直接跟在URL后面2)创建请求对象(默认包含了请求头和请求方法,此步骤可以省略3)创建会话对象(NSURLSession)4)根据会话对象创建请求任务(NSURLSessionDataTask)5)执行Task6)当得到服务器返回的响应后,解析数据(XML|JSON|HTTP)

  • 调用示例 — 采用block的API ①
/* Designated initializer */- (nullable instancetype)initWithRequest:(NSURLRequest *)request delegate:(nullable id)delegate startImmediately:startImmediately API_DEPRECATED("Use NSURLSession (see NSURLSession.h)", macos(10.5,10.11), ios, tvos __WATCHOS_PROHIBITED;- (nullable instancetype)initWithRequest:(NSURLRequest *)request delegate:(nullable id)delegate API_DEPRECATED("Use NSURLSession (see NSURLSession.h)", macos(10.3,10.11), ios, tvos __WATCHOS_PROHIBITED;+ (nullable NSURLConnection*)connectionWithRequest:(NSURLRequest *)request delegate:(nullable id)delegate API_DEPRECATED("Use NSURLSession (see NSURLSession.h)", macos(10.3,10.11), ios, tvos __WATCHOS_PROHIBITED;
  • 代理实现方法示例
/** * 点击按钮 -- 使用NSData下载图片文件,并显示再imageView上 */- downloadBtnClick:(UIButton *)sender { // 在子线程中发送下载文件请求 dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{ // 创建下载路径 NSURL *url = [NSURL URLWithString:@"https://upload-images.jianshu.io/upload_images/1877784-b4777f945878a0b9.jpg"]; // NSData的dataWithContentsOfURL:方法下载 NSData *data = [NSData dataWithContentsOfURL:url]; // 回到主线程,刷新UI dispatch_async(dispatch_get_main_queue(), ^{ self.imageView.image = [UIImage imageWithData:data]; }); });}
#pragma mark <NSURLConnectionDataDelegate> 实现方法/** * 接收到响应的时候:创建一个空的沙盒文件 */- connection:(NSURLConnection *)connection didReceiveResponse:(NSURLResponse *)response{ // 获得下载文件的总长度 self.fileLength = response.expectedContentLength; // 沙盒文件路径 NSString *path = [[NSSearchPathForDirectoriesInDomains(NSCachesDirectory, NSUserDomainMask, YES) lastObject] stringByAppendingPathComponent:@"QQ_V5.4.0.dmg"]; NSLog(@"File downloaded to: %@",path); // 创建一个空的文件到沙盒中 [[NSFileManager defaultManager] createFileAtPath:path contents:nil attributes:nil]; // 创建文件句柄 self.fileHandle = [NSFileHandle fileHandleForWritingAtPath:path];}/** * 接收到具体数据:把数据写入沙盒文件中 */- connection:(NSURLConnection *)connection didReceiveData:data{ // 指定数据的写入位置 -- 文件内容的最后面 [self.fileHandle seekToEndOfFile]; // 向沙盒写入数据 [self.fileHandle writeData:data]; // 拼接文件总长度 self.currentLength += data.length; // 下载进度 self.progressView.progress = 1.0 * self.currentLength / self.fileLength; self.progressLabel.text = [NSString stringWithFormat:@"当前下载进度:%.2f%%",100.0 * self.currentLength / self.fileLength];}/** * 下载完文件之后调用:关闭文件、清空长度 */- connectionDidFinishLoading:(NSURLConnection *)connection{ // 关闭fileHandle [self.fileHandle closeFile]; self.fileHandle = nil; // 清空长度 self.currentLength = 0; self.fileLength = 0;}

//handler A block which receives the results of the resource load.+ sendAsynchronousRequest:(NSURLRequest*) request queue:(NSOperationQueue*) queue completionHandler:(NSURLResponse* _Nullable response, NSData* _Nullable data, NSError* _Nullable connectionError)) handler API_DEPRECATED("Use [NSURLSession dataTaskWithRequest:completionHandler:] (see NSURLSession.h", macos(10.7,10.11), ios, tvos __WATCHOS_PROHIBITED;

② 下载完的事件采用delegate形式
  • 关键API

requestWithURL:sessionWithConfigurationdataTaskWithRequest:request

  • get请求发起示例

- getWithDelegate1{ //1.确定请求路径 NSURL *url = [NSURL URLWithString:@"http://120.25.226.186:32812/login?username=520it&pwd=520it&type=JSON"]; //2.创建请求对象 //请求对象内部默认已经包含了请求头和请求方法 NSURLRequest *request = [NSURLRequest requestWithURL:url]; //3.获得会话对象,并设置代理 /* 第一个参数:会话对象的配置信息defaultSessionConfiguration 表示默认配置 第二个参数:谁成为代理,此处为控制器本身即self 第三个参数:队列,该队列决定代理方法在哪个线程中调用,可以传主队列|非主队列 [NSOperationQueue mainQueue] 主队列: 代理方法在主线程中调用 [[NSOperationQueue alloc]init] 非主队列: 代理方法在子线程中调用 */ NSURLSession *session = [NSURLSession sessionWithConfiguration:[NSURLSessionConfiguration defaultSessionConfiguration] delegate:self delegateQueue:[NSOperationQueue mainQueue]]; //4.根据会话对象创建一个Task NSURLSessionDataTask *dataTask = [session dataTaskWithRequest:request]; //5.执行任务 [dataTask resume];}
  • 代理关键API

//1.接收到服务器响应的时候调用该方法-URLSession:(NSURLSession *)session dataTask:(NSURLSessionDataTask *)dataTask didReceiveResponse:(NSURLResponse *)response completionHandler:(NSURLSessionResponseDisposition))completionHandler{//2.接收到服务器返回数据的时候会调用该方法,如果数据较大那么该方法可能会调用多次-URLSession:(NSURLSession *)session dataTask:(NSURLSessionDataTask *)dataTask didReceiveData:data{//3.当请求完成的时候会调用该方法,如果请求失败,则error有值-URLSession:(NSURLSession *)session task:(NSURLSessionTask *)task didCompleteWithError:(NSError *)error{
  • 代理实现示例

#pragma mark NSURLSessionDataDelegate//1.接收到服务器响应的时候调用该方法-URLSession:(NSURLSession *)session dataTask:(NSURLSessionDataTask *)dataTask didReceiveResponse:(NSURLResponse *)response completionHandler:(NSURLSessionResponseDisposition))completionHandler{ //在该方法中可以得到响应头信息,即response NSLog(@"didReceiveResponse--%@",[NSThread currentThread]); //注意:需要使用completionHandler回调告诉系统应该如何处理服务器返回的数据 //默认是取消的 /* NSURLSessionResponseCancel = 0, 默认的处理方式,取消 NSURLSessionResponseAllow = 1, 接收服务器返回的数据 NSURLSessionResponseBecomeDownload = 2,变成一个下载请求 NSURLSessionResponseBecomeStream 变成一个流 */ completionHandler(NSURLSessionResponseAllow);}//2.接收到服务器返回数据的时候会调用该方法,如果数据较大那么该方法可能会调用多次-URLSession:(NSURLSession *)session dataTask:(NSURLSessionDataTask *)dataTask didReceiveData:data{ NSLog(@"didReceiveData--%@",[NSThread currentThread]); //拼接服务器返回的数据 [self.responseData appendData:data];}//3.当请求完成的时候会调用该方法,如果请求失败,则error有值-URLSession:(NSURLSession *)session task:(NSURLSessionTask *)task didCompleteWithError:(NSError *)error{ NSLog(@"didCompleteWithError--%@",[NSThread currentThread]); if(error == nil) { //解析数据,JSON解析请参考http://www.cnblogs.com/wendingding/p/3815303.html NSDictionary *dict = [NSJSONSerialization JSONObjectWithData:self.responseData options:kNilOptions error:nil]; NSLog(@"%@",dict); }}
  • 如下图所示,get请求内部封装了开始操作,不用再像NSURLSession一样外面resume了。

    ca88会员登录中心 3

  • post请求示例
  • 关键API

sharedSessionrequestWithURL:request.HTTPMethod =
@”POST”;dataTaskWithRequest:request completionHandler:

-postWithBlock{ //对请求路径的说明 //http://120.25.226.186:32812/login //协议头+主机地址+接口名称 //协议头+主机地址(120.25.226.186:32812)+接口名称 //POST请求需要修改请求方法为POST,并把参数转换为二进制数据设置为请求体 //1.创建会话对象 NSURLSession *session = [NSURLSession sharedSession]; //2.根据会话对象创建task NSURL *url = [NSURL URLWithString:@"http://120.25.226.186:32812/login"]; //3.创建可变的请求对象 NSMutableURLRequest *request = [NSMutableURLRequest requestWithURL:url]; //4.修改请求方法为POST request.HTTPMethod = @"POST"; //5.设置请求体 request.HTTPBody = [@"username=520it&pwd=520it&type=JSON" dataUsingEncoding:NSUTF8StringEncoding]; //6.根据会话对象创建一个Task /* 第一个参数:请求对象 第二个参数:completionHandler回调(请求完成的回调) data:响应体信息 response:响应头信息,主要是对服务器端的描述 error:错误信息,如果请求失败,则error有值 */ NSURLSessionDataTask *dataTask = [session dataTaskWithRequest:request completionHandler:^(NSData * _Nullable data, NSURLResponse * _Nullable response, NSError * _Nullable error) { //8.解析数据 NSDictionary *dict = [NSJSONSerialization JSONObjectWithData:data options:kNilOptions error:nil]; NSLog(@"%@",dict); }]; //7.执行任务 [dataTask resume];}

前面涉及到的GET和POST都属于HTTP请求,现在苹果的APP都推荐支持HTTPS,这就需要先配置一下证书,然后在NSURLSession(或者NSURLConnection但现在新的项目基本不用了)的代理方法里面进行一些特别的操作。如果是AFNetWorking,也需要对AFHTTPRequestOperationManager对象进行一些特别的操作。

关于证书的配置,及需要的特别的操作,推荐阅读:

AFNetworking2.0和3.0区别很大,也是因为苹果废弃了NSURLConnection,而改用了NSURLSession,AFNetworking3.0实际上只是对NSURLSession所做的操作进行了高度封装,提供更加简洁的API供编码调用。

查看AFHTTPSessionManager.h文件,可知AFHTTPSessionManager是AFURLSessionManager的子类:

@interface AFHTTPSessionManager : AFURLSessionManager <NSSecureCoding, NSCopying>

请求示例 — 下载一个PDF文件

- DownloadPdfAndSave{ AFHTTPSessionManager *manager = [AFHTTPSessionManager manager]; manager.responseSerializer = [AFHTTPResponseSerializer serializer]; manager.responseSerializer.acceptableContentTypes = [NSSet setWithObject:@"application/pdf"]; __weak __typeof__ weakSelf = self; //临时配置,需要自己根据接口地址改动!!!!!!!!!!!!!!!!!!!! self.urlStr = @"http://10.20.201.78/test3.pdf"; [manager GET:_urlStr parameters:nil progress:^(NSProgress * _Nonnull downloadProgress) { } success:^(NSURLSessionDataTask * _Nonnull task, id _Nullable responseObject) { __strong __typeof__ strongSelf = weakSelf; strongSelf.isWriten = [responseObject writeToFile:[self pathOfPdf] atomically:YES]; [strongSelf openPdfByAddingSubView]; //[strongSelf.previewController reloadData]; } failure:^(NSURLSessionDataTask * _Nullable task, NSError * _Nonnull error) { NSLog(@"%@",error.userInfo); }];}

get请求调用栈分析

  • AFHTTPSessionManager.m

- (NSURLSessionDataTask *)GET:(NSString *)URLString parameters:parameters success:(NSURLSessionDataTask *task, id responseObject))success failure:(NSURLSessionDataTask *task, NSError *error))failure{ return [self GET:URLString parameters:parameters progress:nil success:success failure:failure];}

- (NSURLSessionDataTask *)GET:(NSString *)URLString parameters:parameters progress:(NSProgress * _Nonnull))downloadProgress success:(NSURLSessionDataTask * _Nonnull, id _Nullable))success failure:(NSURLSessionDataTask * _Nullable, NSError * _Nonnull))failure{ NSURLSessionDataTask *dataTask = [self dataTaskWithHTTPMethod:@"GET" URLString:URLString parameters:parameters uploadProgress:nil downloadProgress:downloadProgress success:success failure:failure]; [dataTask resume]; return dataTask;}

- (NSURLSessionDataTask *)dataTaskWithHTTPMethod:(NSString *)method URLString:(NSString *)URLString parameters:parameters uploadProgress:(nullable void (NSProgress *uploadProgress)) uploadProgress downloadProgress:(nullable void (NSProgress *downloadProgress)) downloadProgress success:(NSURLSessionDataTask *, id))success failure:(NSURLSessionDataTask *, NSError *))failure{ NSError *serializationError = nil; NSMutableURLRequest *request = [self.requestSerializer requestWithMethod:method URLString:[[NSURL URLWithString:URLString relativeToURL:self.baseURL] absoluteString] parameters:parameters error:&serializationError]; if (serializationError) { if  { dispatch_async(self.completionQueue ?: dispatch_get_main_queue(), ^{ failure(nil, serializationError); }); } return nil; } __block NSURLSessionDataTask *dataTask = nil; dataTask = [self dataTaskWithRequest:request uploadProgress:uploadProgress downloadProgress:downloadProgress completionHandler:^(NSURLResponse * __unused response, id responseObject, NSError *error) { if  { if  { failure(dataTask, error); } } else { if  { success(dataTask, responseObject); } } }]; return dataTask;}
  • AFURLSessionManager.m

- (NSURLSessionDataTask *)dataTaskWithRequest:(NSURLRequest *)request uploadProgress:(nullable void (NSProgress *uploadProgress)) uploadProgressBlock downloadProgress:(nullable void (NSProgress *downloadProgress)) downloadProgressBlock completionHandler:(nullable void (NSURLResponse *response, id _Nullable responseObject, NSError * _Nullable error))completionHandler { __block NSURLSessionDataTask *dataTask = nil; url_session_manager_create_task_safely(^{ dataTask = [self.session dataTaskWithRequest:request]; }); [self addDelegateForDataTask:dataTask uploadProgress:uploadProgressBlock downloadProgress:downloadProgressBlock completionHandler:completionHandler]; return dataTask;}
  • AFURLSessionManager.m

调用示例DownloadVC.m

- downloadBtnClicked:(UIButton *)sender { NSURLSessionConfiguration *configuration = [NSURLSessionConfiguration defaultSessionConfiguration]; // 1. 创建会话管理者 AFURLSessionManager *manager = [[AFURLSessionManager alloc] initWithSessionConfiguration:configuration]; // 2. 创建下载路径和请求对象 NSURL *URL = [NSURL URLWithString:@"http://dldir1.qq.com/qqfile/QQforMac/QQ_V5.4.0.dmg"]; NSURLRequest *request = [NSURLRequest requestWithURL:URL]; // 3.创建下载任务 /** * 第一个参数 - request:请求对象 * 第二个参数 - progress:下载进度block * 其中: downloadProgress.completedUnitCount:已经完成的大小 * downloadProgress.totalUnitCount:文件的总大小 * 第三个参数 - destination:自动完成文件剪切操作 * 其中: 返回值:该文件应该被剪切到哪里 * targetPath:临时路径 tmp NSURL * response:响应头 * 第四个参数 - completionHandler:下载完成回调 * 其中: filePath:真实路径 == 第三个参数的返回值 * error:错误信息 */ NSURLSessionDownloadTask *downloadTask = [manager downloadTaskWithRequest:request progress:^(NSProgress *downloadProgress) { __weak typeof weakSelf = self; // 获取主线程,不然无法正确显示进度。 NSOperationQueue* mainQueue = [NSOperationQueue mainQueue]; [mainQueue addOperationWithBlock:^{ // 下载进度 weakSelf.progressView.progress = 1.0 * downloadProgress.completedUnitCount / downloadProgress.totalUnitCount; weakSelf.progressLabel.text = [NSString stringWithFormat:@"当前下载进度:%.2f%%",100.0 * downloadProgress.completedUnitCount / downloadProgress.totalUnitCount]; }]; } destination:^NSURL *(NSURL *targetPath, NSURLResponse *response) { NSURL *path = [[NSFileManager defaultManager] URLForDirectory:NSDocumentDirectory inDomain:NSUserDomainMask appropriateForURL:nil create:NO error:nil]; return [path URLByAppendingPathComponent:@"QQ_V5.4.0.dmg"]; } completionHandler:^(NSURLResponse *response, NSURL *filePath, NSError *error) { NSLog(@"File downloaded to: %@", filePath); }]; // 4. 开启下载任务 [downloadTask resume];}

内部封装分析AFURLSessionManager.m

- (NSURLSessionDownloadTask *)downloadTaskWithRequest:(NSURLRequest *)request progress:(NSProgress *downloadProgress)) downloadProgressBlock destination:(NSURL * (NSURL *targetPath, NSURLResponse *response))destination completionHandler:(NSURLResponse *response, NSURL *filePath, NSError *error))completionHandler{ __block NSURLSessionDownloadTask *downloadTask = nil; url_session_manager_create_task_safely(^{ downloadTask = [self.session downloadTaskWithRequest:request]; }); [self addDelegateForDownloadTask:downloadTask progress:downloadProgressBlock destination:destination completionHandler:completionHandler]; return downloadTask;}

其中self.session是AFURLSessionManager.h中的属性

@property (readonly, nonatomic, strong) NSURLSession *session;

它后面调用的API声明在NSFoundation的NSURLSession.h的头文件中

/* Creates a data task with the given request. The request may have a body stream. */- (NSURLSessionDataTask *)dataTaskWithRequest:(NSURLRequest *)request;

添加代理的封装AFURLSessionManager.m

- addDelegateForDataTask:(NSURLSessionDataTask *)dataTask uploadProgress:(nullable void (NSProgress *uploadProgress)) uploadProgressBlock downloadProgress:(nullable void (NSProgress *downloadProgress)) downloadProgressBlock completionHandler:(NSURLResponse *response, id responseObject, NSError *error))completionHandler{ AFURLSessionManagerTaskDelegate *delegate = [[AFURLSessionManagerTaskDelegate alloc] init]; delegate.manager = self; delegate.completionHandler = completionHandler; dataTask.taskDescription = self.taskDescriptionForSessionTasks; [self setDelegate:delegate forTask:dataTask]; delegate.uploadProgressBlock = uploadProgressBlock; delegate.downloadProgressBlock = downloadProgressBlock;}

其中

- setDelegate:(AFURLSessionManagerTaskDelegate *)delegate forTask:(NSURLSessionTask *)task{ NSParameterAssert; NSParameterAssert; [self.lock lock]; self.mutableTaskDelegatesKeyedByTaskIdentifier[@(task.taskIdentifier)] = delegate; [delegate setupProgressForTask:task]; [self addNotificationObserverForTask:task]; [self.lock unlock];}

其中,self.mutableTaskDelegatesKeyedByTaskIdentifier是个字典

@property (readwrite, nonatomic, strong) NSMutableDictionary *mutableTaskDelegatesKeyedByTaskIdentifier;

被调用的地方在:

- (AFURLSessionManagerTaskDelegate *)delegateForTask:(NSURLSessionTask *)task { NSParameterAssert; AFURLSessionManagerTaskDelegate *delegate = nil; [self.lock lock]; delegate = self.mutableTaskDelegatesKeyedByTaskIdentifier[@(task.taskIdentifier)]; [self.lock unlock]; return delegate;}

进而被调用的地方在:

- URLSession:(NSURLSession *)session downloadTask:(NSURLSessionDownloadTask *)downloadTaskdidFinishDownloadingToURL:location{ AFURLSessionManagerTaskDelegate *delegate = [self delegateForTask:downloadTask]; if (self.downloadTaskDidFinishDownloading) { NSURL *fileURL = self.downloadTaskDidFinishDownloading(session, downloadTask, location); if  { delegate.downloadFileURL = fileURL; NSError *error = nil; [[NSFileManager defaultManager] moveItemAtURL:location toURL:fileURL error:&error]; if  { [[NSNotificationCenter defaultCenter] postNotificationName:AFURLSessionDownloadTaskDidFailToMoveFileNotification object:downloadTask userInfo:error.userInfo]; } return; } } if  { [delegate URLSession:session downloadTask:downloadTask didFinishDownloadingToURL:location]; }}

说明:这个NSURLSession的API容易跟AFURLSessionManager的API混淆,参数都是一个request和一个handler
block。

  • NSURLSession的API是这样的:

- (NSURLSessionDataTask *)dataTaskWithRequest:(NSURLRequest *)request completionHandler:(NSData * _Nullable data, NSURLResponse * _Nullable response, NSError * _Nullable error))completionHandler{
  • AFURLSessionManager的API是这样的,可以对比学习下:

- (NSURLSessionDataTask *)dataTaskWithRequest:(NSURLRequest *)request completionHandler:(NSURLResponse *response, id responseObject, NSError *error))completionHandler{

调用示例 — dataTaskWithRequest:DownloadVC.m

 // 创建下载URL NSURL *url = [NSURL URLWithString:@"http://dldir1.qq.com/qqfile/QQforMac/QQ_V5.4.0.dmg"]; // 2.创建request请求 NSMutableURLRequest *request = [NSMutableURLRequest requestWithURL:url]; // 设置HTTP请求头中的Range NSString *range = [NSString stringWithFormat:@"bytes=%zd-", self.currentLength]; [request setValue:range forHTTPHeaderField:@"Range"]; __weak typeof weakSelf = self; _downloadTask = [self.manager dataTaskWithRequest:request completionHandler:^(NSURLResponse * _Nonnull response, id _Nullable responseObject, NSError * _Nullable error) { NSLog(@"dataTaskWithRequest"); // 清空长度 weakSelf.currentLength = 0; weakSelf.fileLength = 0; // 关闭fileHandle [weakSelf.fileHandle closeFile]; weakSelf.fileHandle = nil; }];

其中self.manager是懒加载得到的AFURLSessionManager

/** * manager的懒加载 */- (AFURLSessionManager *)manager { if (!_manager) { NSURLSessionConfiguration *configuration = [NSURLSessionConfiguration defaultSessionConfiguration]; // 1. 创建会话管理者 _manager = [[AFURLSessionManager alloc] initWithSessionConfiguration:configuration]; } return _manager;}

内部封装分析AFURLSessionManager.m

- (NSURLSessionDataTask *)dataTaskWithRequest:(NSURLRequest *)request completionHandler:(NSURLResponse *response, id responseObject, NSError *error))completionHandler{ return [self dataTaskWithRequest:request uploadProgress:nil downloadProgress:nil completionHandler:completionHandler];}

- (NSURLSessionDataTask *)dataTaskWithRequest:(NSURLRequest *)request uploadProgress:(nullable void (NSProgress *uploadProgress)) uploadProgressBlock downloadProgress:(nullable void (NSProgress *downloadProgress)) downloadProgressBlock completionHandler:(nullable void (NSURLResponse *response, id _Nullable responseObject, NSError * _Nullable error))completionHandler { __block NSURLSessionDataTask *dataTask = nil; url_session_manager_create_task_safely(^{ dataTask = [self.session dataTaskWithRequest:request]; }); [self addDelegateForDataTask:dataTask uploadProgress:uploadProgressBlock downloadProgress:downloadProgressBlock completionHandler:completionHandler]; return dataTask;}

初始化AFHTTPSessionManager的内部实现调用栈

  • [AFHTTPSessionManager initWithBaseURL:]
    • [AFHTTPSessionManager initWithBaseURL:sessionConfiguration:]
      • [AFURLSessionManager initWithSessionConfiguration:] //
        调用了父类AFURLSessionManager的初始化方法

        • [NSURLSession
          sessionWithConfiguration:delegate:delegateQueue:] //
          调用了原生类NSURLSession的初始化方法
        • [AFJSONResponseSerializer serializer]
        • [AFSecurityPolicy defaultPolicy]
        • [AFNetworkReachabilityManager sharedManager]
      • [AFHTTPRequestSerializer serializer]
      • [AFJSONResponseSerializer serializer]

AFHTTPSessionManager发送请求的内部实现调用栈

  • [AFHTTPSessionManager GET:parameters:process:success:failure:]
    • [AFHTTPSessionManager
      dataTaskWithHTTPMethod:parameters:uploadProgress:downloadProgress:success:failure:]
      //

      • [AFHTTPRequestSerializer
        requestWithMethod:URLString:parameters:error:] //
        获得NSMutableURLRequest
      • [AFURLSessionManager
        dataTaskWithRequest:uploadProgress:downloadProgress:completionHandler:]
        //

        • [NSURLSession dataTaskWithRequest:] //
        • [AFURLSessionManager
          addDelegateForDataTask:uploadProgress:downloadProgress:completionHandler:]
          // 添加代理

          • [AFURLSessionManagerTaskDelegate init]
          • [AFURLSessionManager setDelegate:forTask:]
    • [NSURLSessionDataTask resume]

其中,、、这三个方法得到的是同一个对象,即中系统原生的NSURLSessionDataTask对象。所以,AF请求操作内部实现也是和原生NSURLSession操作一样,创建task,调用resume发送请求。

请求的时候,NSURLSession的session跟TCP的个数是否有什么关系?有人说请求同域名且共享的session会复用同一个TCP链接,否则就不复用,就一个session一个TCP连接?

关于这块的知识可研究资料较少,且不可信,笔者日后研究到确定的答案后再更新。也欢迎读者留下自己的见解。

不过据我观察,可能没那么简单,新的iOS11系统新增了多路TCP即Multipath-TCP,因而也为NSURLSession和NSURLSessionConfiguration提供了新的属性multipathServiceType,以及HTTPMaximumConnectionsPerHost。下面是它们的定义:

  • NSURLSession.h

/* multipath service type to use for connections. The default is NSURLSessionMultipathServiceTypeNone */@property NSURLSessionMultipathServiceType multipathServiceType API_AVAILABLE) API_UNAVAILABLE(macos, watchos, tvos);/* The maximum number of simultanous persistent connections per host */@property NSInteger HTTPMaximumConnectionsPerHost;
  • NSURLSession.h

typedef NS_ENUM(NSInteger, NSURLSessionMultipathServiceType){ NSURLSessionMultipathServiceTypeNone = 0, /* None - no multipath  */ NSURLSessionMultipathServiceTypeHandover = 1, /* Handover - secondary flows brought up when primary flow is not performing adequately. */ NSURLSessionMultipathServiceTypeInteractive = 2, /* Interactive - secondary flows created more aggressively. */ NSURLSessionMultipathServiceTypeAggregate = 3 /* Aggregate - multiple subflows used for greater bandwitdh. */} API_AVAILABLE) API_UNAVAILABLE(macos, watchos, tvos) NS_SWIFT_NAME(URLSessionConfiguration.MultipathServiceType);
  • NSURLSessionConfiguration.h

/* multipath service type to use for connections. The default is NSURLSessionMultipathServiceTypeNone */@property NSURLSessionMultipathServiceType multipathServiceType API_AVAILABLE) API_UNAVAILABLE(macos, watchos, tvos);/* The maximum number of simultanous persistent connections per host */@property NSInteger HTTPMaximumConnectionsPerHost;
  • 关键步骤NSString–>NSURL–>NSData–>UIImage
  • 关键API

URLWithStringdataWithContentsOfURL:urlimageWithData:data

发表评论

电子邮件地址不会被公开。 必填项已用*标注

网站地图xml地图