标签:
有关网络请求的类(该图片来自:developer.apple.com)
网络请求的组成部分有服务器地址、请求参数,以及请求方式。在iOS中,一个网络请求用NSURLRequest(或者其子类NSURLMutableRequest)来表示。
NSURLRequest适合于get请求,NSURLMutableRequest可以是post请求。
get方式下,服务器地址和请求参数都可以明文包含在url中,可以直接使用这个url来创建NSURLRequest(或者NSURLMutableRequest)。
post方式下,服务器地址用NSURL来表示,请求参数可以封装到NSData中(也可以以文件或者流的形式保存)。由于需要将服务器地址和请求参数分开,所以需要用到NSURLMutableRequest类。用服务器地址的url创建NSURLMutableRequest对象,然后将封装了请求参数的NSData对象添加到这个NSURLMutableRequest对象中。示例如下:
NSURL *url=[NSURL URLWithString:@“http://ipad-bjwb.bjd.com.cn/DigitalPublication/publish/Handler/APINewsList.ashx"];
NSData *data=[@"date=20151101&startRecord=1&len=5&udid=1234567890&terminalType=Iphone&cid=213" dataUsingEncoding:NSUTF8StringEncoding];
然后将服务器地址与请求参数加入到NSURLMutableRequest对象中:
NSMutableURLRequest *request=[NSMutableURLRequest requestWithURL:url cachePolicy:0 timeoutInterval:60];
request.HTTPMethod=@"post";//指明请求方式为post,默认为get
request.HTTPBody=data;
这样就准备好了一个网络请求,接下来就是如何去执行这个网络请求的问题。
执行url请求可以使用iOS原生API也可以使用第三方库。
执行url请求的两个原生方法:
(1)NSURLSession
(2)NSURLConnection
iOS 7.0 以后推荐使用NSURLSession,以前使用NSURLConnection。
NSURLSession、NSURLConnection都可以发送和接收以NSData或者文件的形式保存的数据.
实现url请求的第三方库有afNetworking、sdWebImage等。
(1)每个项目都有一个NSURLSession对象,可以通过[NSURLSession sharedSession]方法来获取它,然后就可以用它来新建请求任务。该任务调用resume方法后开始执行。请求任务有三种,上传、下载、数据。通过NSURLSession来执行url请求是异步的。
(2)NSURLConnection直接执行send方法(类方法)发送请求,可以选择同步也可以选择异步。异步发送请求接收数据的话可以使用block接收,也可以使用代理。注意如果使用代理来接收数据,数据会是断续接收的。通常使用block比较方便。
同步和异步的区别就是,同步就是执行完网络请求方法后,要等到网络请求得到响应并完成数据传输之后才返回,异步就是发送完请求之后马上返回。
获取session可以通过初始化方法,也可以直接使用项目默认的session对象,通过sharedSession方法获取。初始化session对象需要指定代理,如果指定为nil,就是使用系统默认的代理。
session分为3种:
初始化session的时候需要指定一个NSURLSessionConfiguration对象,session的类型就由这个对象的类型来确定。
任务分为3种,数据、下载、上传:
任务的初始化和任务的管理都由session来完成。
每个session对象管理着它自己创建的任务。
任务创建后需要执行resume方法,session才会处理它。
任务处理的具体操作是由session的代理来完成的,也就是说具体的处理会写在代理方法中。
需要注意的问题:
后台任务与app的交互:
后台任务的特点就是运行在后台,当app不在运行的时候后台任务仍然可以运行。
上面说过,任务的处理是由session的代理方法来完成的,但是当app没有运行的时候,session已经被释放,这时候后台任务怎么和创建这个任务的session重新关联起来呢?
这个问题iOS系统已经有一套完整的机制去处理它。是这样的:当后台任务完成或者需要认证的时候,如果app没有在运行,系统就会唤醒app,并调用app delegate的(void)application:(UIApplication *)application handleEventsForBackgroundURLSession:(NSString *)identifier completionHandler:(void (^)())completionHandler
方法。这个方法就是特地写来处理后台任务的,下面所描述的内容都围绕这个方法来展开。
需要重写在这个方法,在这个方法中让后台任务和session重新关联起来。由于创建它的session已经释放,要找回它不可能,实际的做法是创建一个和它一样的session,然后让任务与这个session关联起来。具体就是用这个方法的参数identifier来新建一个NSURLSessionConfiguration对象,并且用这个NSURLSessionConfiguration对象新建一个session,这个session就可以当成是原来的session了。神奇的是,只要这样创建了session,正在运行的任务就会自动和它关联起来,至于是怎么做到了,不需要操心,系统已经解决了。
留意到这个方法有一个block类型的参数completionHandler,它的意义在于,一旦它被执行就表明后台任务处理完成,因此系统自唤醒app之后就会一直等待completionHandler的执行,以便让app重新停止运行。因此,要记得把completionHandler传给session的代理,并且在session的代理方法URLSessionDidFinishEventsForBackgroundURLSession:中调用它,这样就实现了在最后一个任务处理完毕以后让app再次进入后台。
在正在运行的任务已经和session重新取得联系之后就会调用其代理的相关方法,如果是后台任务完成,就会调用URLSession:downloadTask:didFinishDownloadingToURL:方法。如果是需要认证,就会调用URLSession:task:didReceiveChallenge:completionHandler:方法或者URLSession:didReceiveChallenge:completionHandler:方法。
与文件下载有关的协议方法:
URLSession:downloadTask:didFinishDownloadingToURL:
URLSession:downloadTask:didWriteData:totalBytesWritten:totalBytesExpectedToWrite:
URLSession:downloadTask:didResumeAtOffset:expectedTotalBytes:
URLSession:task:didCompleteWithError:
NSURLSessionDownloadTask
对象的方法:cancelByProducingResumeData:
session
的downloadTaskWithResumeData:
或downloadTaskWithResumeData:completionHandler:
URLSession:task:didCompleteWithError:
中的NSError对象的userInfo字典中会有,对应key为NSURLSessionDownloadTaskResumeData
。相关协议方法:
- (void)URLSession:(NSURLSession *)session? task:(NSURLSessionTask *)task?willPerformHTTPRedirection:(NSHTTPURLResponse *)response? newRequest:(NSURLRequest *)request? completionHandler:(void (^)(NSURLRequest *))completionHandler
对于默认的session和暂时的session管理的url请求任务发生重定向时就会调用这个方法,而后台任务发生重定向会直接服从重定向。
在这个方法里面,可以获取到新的url请求,调用completionHandler(request);
就会允许重定向。如果调用completionHandler(nil);
就相当于拒绝重定向。当然也可以调用completionHandler(otherRequest);
实现自定义重定向。
标签:
原文地址:http://blog.csdn.net/ohyeahhhh/article/details/51338273