iOS开发网络篇—发送GET和POST请求(使用NSURLSession) - 转
说明:
1.该文主要介绍如何使用nsurlsession来发送get请求和post请求
2.本文将不再讲解nsurlconnection的使用,如有需要了解nsurlconnection如何发送请求。
详细信息,请参考:http://www.cnblogs.com/wendingding/p/3813706.html
3.本文示例代码发送的请求均为http请求,已经对info.plist文件进行配置。
如何配置,请参考:https://github.com/hangangandhanmeimei/ios9adaptationtips
4.本文示例代码,可以在下面的地址获取:
https://github.com/hangangandhanmeimei/code
一、简单说明
在ios9.0之后,以前使用的nsurlconnection过期,苹果推荐使用nsurlsession来替换nsurlconnection完成网路请求相关操作。
nsurlsession的使用非常简单,先根据会话对象创建一个请求task,然后执行该task即可。
nsurlsessiontask本身是一个抽象类,在使用的时候,通常是根据具体的需求使用它的几个子类。关系如下:
二、发送get请求
使用nsurlsession发送get请求的方法和nsurlconnection类似,整个过程如下:
1.确定请求路径(一般由公司的后台开发人员以接口文档的方式提供),get请求参数直接跟在url后面
2.创建请求对象(默认包含了请求头和请求方法【get】),此步骤可以省略
3.创建会话对象(nsurlsession)
4.根据会话对象创建请求任务(nsurlsessiondatatask)
5.执行task
6.当得到服务器返回的响应后,解析数据(xml|json|http)
示例代码:
-(void)get1 { //对请求路径的说明 //http://120.25.226.186:32812/login?username=520it&pwd=520&type=json //协议头+主机地址+接口名称+?+参数1&参数2&参数3 //协议头(http://)+主机地址(120.25.226.186:32812)+接口名称(login)+?+参数1(username=520it)&参数2(pwd=520)&参数3(type=json) //get请求,直接把请求参数跟在url的后面以?隔开,多个参数之间以&符号拼接 //1.确定请求路径 nsurl *url = [nsurl urlwithstring:@"http://120.25.226.186:32812/login?username=520it&pwd=520it&type=json"]; //2.创建请求对象 //请求对象内部默认已经包含了请求头和请求方法(get) 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请求的第一种方法
-(void)get2 { //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)该方法内部会自动将请求路径包装成一个请求对象,该请求对象默认包含了请求头信息和请求方法(get) 2)如果要发送的是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]; } 发送get请求的第二种方法 执行结果 ![](http://images2015.cnblogs.com/blog/450136/201601/450136-20160129131707286-1497362695.png)
此处打印的值是一个字典,字典中success这个key对应的value打印出来为unicode编码的,如果想输出中文,可以为nsdictionary提供一个分类,重写系统中的方法
#import "nsdictionary+log.h" @implementation nsdictionary (log) -(nsstring *)descriptionwithlocale:(id)locale indent:(nsuinteger)level { //初始化可变字符串 nsmutablestring *string = [nsmutablestring string]; //拼接开头[ [string appendstring:@"["]; //拼接字典中所有的键值对 [self enumeratekeysandobjectsusingblock:^(id _nonnull key, id _nonnull obj, bool * _nonnull stop) { [string appendformat:@"%@:",key]; [string appendformat:@"%@",obj]; }]; //拼接结尾] [string appendstring:@"]"]; return string; } @end 字典分类中重写系统方法
执行结果:
三、发送post请求
使用nsurlsession发送post请求的方法和nsurlconnection类似,整个过程如下:
1)确定请求路径(一般由公司的后台开发人员以接口文档的方式提供)
2)创建可变的请求对象(因为需要修改),此步骤不可以省略
3)修改请求方法为post
4)设置请求体,把参数转换为二进制数据并设置请求体
5)创建会话对象(nsurlsession)
6)根据会话对象创建请求任务(nsurlsessiondatatask)
7)执行task
8)当得到服务器返回的响应后,解析数据(xml|json|http)
示例代码:
-(void)post { //对请求路径的说明 //http://120.25.226.186:32812/login //协议头+主机地址+接口名称 //协议头(http://)+主机地址(120.25.226.186:32812)+接口名称(login) //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]; } 发送post请求的方法
四、nsurlsession代理方法简单介绍
有的时候,我们可能需要监听网络请求的过程(如下载文件需监听文件下载进度),那么就需要用到代理方法。
接下来通过代码简单说明nsurlsession中普通网络请求会涉及代理方法的使用
#import "viewcontroller.h" @interface viewcontroller ()<nsurlsessiondatadelegate> @property (nonatomic, strong) nsmutabledata *responsedata; @end @implementation viewcontroller -(nsmutabledata *)responsedata { if (_responsedata == nil) { _responsedata = [nsmutabledata data]; } return _responsedata; } //当点击控制器view的时候会调用该方法 -(void)touchesbegan:(nsset<uitouch *> *)touches withevent:(uievent *)event { [self delegatetest]; } //发送请求,代理方法 -(void)delegatetest { //1.确定请求路径 nsurl *url = [nsurl urlwithstring:@"http://120.25.226.186:32812/login?username=520it&pwd=520it&type=json"]; //2.创建请求对象 //请求对象内部默认已经包含了请求头和请求方法(get) 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]; } //1.接收到服务器响应的时候调用该方法 -(void)urlsession:(nsurlsession *)session datatask:(nsurlsessiondatatask *)datatask didreceiveresponse:(nsurlresponse *)response completionhandler:(void (^)(nsurlsessionresponsedisposition))completionhandler { //在该方法中可以得到响应头信息,即response nslog(@"didreceiveresponse--%@",[nsthread currentthread]); //注意:需要使用completionhandler回调告诉系统应该如何处理服务器返回的数据 //默认是取消的 /* nsurlsessionresponsecancel = 0, 默认的处理方式,取消 nsurlsessionresponseallow = 1, 接收服务器返回的数据 nsurlsessionresponsebecomedownload = 2,变成一个下载请求 nsurlsessionresponsebecomestream 变成一个流 */ completionhandler(nsurlsessionresponseallow); } //2.接收到服务器返回数据的时候会调用该方法,如果数据较大那么该方法可能会调用多次 -(void)urlsession:(nsurlsession *)session datatask:(nsurlsessiondatatask *)datatask didreceivedata:(nsdata *)data { nslog(@"didreceivedata--%@",[nsthread currentthread]); //拼接服务器返回的数据 [self.responsedata appenddata:data]; } //3.当请求完成(成功|失败)的时候会调用该方法,如果请求失败,则error有值 -(void)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); } } @end
执行结果:
原文转自