iOS NSURLConnection使用详解
一、整体介绍
NSURLConnection是苹果提供的原生网络访问类,但是苹果很快会将其废弃,且由NSURLSession(iOS7以后)来替代。目前使用最广泛的第三方网络框架AFNetworking最新版本已弃用了NSURLConnection,那我们学习它还有什么用呢?
- 首先,苹果弃用它还是需要时间的,最起码到iOS10之后;
- 现在还有一些老项目会使用NSURLConnection,特别是2013年之前的项目,用户量基础还是很大的;
- 另外,不得不承认,有些公司还在用类似ASI这些经典的网络框架,所以还是很有必要学习NSURLConnection的。
二、使用的一般步骤
1 NSURL:请求地址,定义一个网络资源路径:
NSURL *url = [NSURL URLWithString:@"协议://主机地址/路径?参数&参数"];
解释如下:
- 协议:不同的协议,代表着不同的资源查找方式、资源传输方式,比如常用的http,ftp等
- 主机地址:存放资源的主机的IP地址(域名)
- 路径:资源在主机中的具体位置
- 参数:参数可有可无,也可以多个。如果带参数的话,用“?”号后面接参数,多个参数的话之间用&隔开
2 NSURLRequest:请求,根据前面的NSURL建立一个请求:
NSURLRequest *request = [NSURLRequest requestWithURL:url cachePolicy:NSURLRequestUseProtocolCachePolicy timeoutInterval:15.0];
参数解释如下:
- url:资源路径
- cachePolicy:缓存策略(无论使用哪种缓存策略,都会在本地缓存数据),类型为美剧类型,取值如下:
- NSURLRequestUseProtocolCachePolicy = 0 //默认的缓存策略,使用协议的缓存策略
- NSURLRequestReloadIgnoringLocalCacheData = 1 //每次都从网络加载
- NSURLRequestReturnCacheDataElseLoad = 2 //返回缓存否则加载,很少使用
- NSURLRequestReturnCacheDataDontLoad = 3 //只返回缓存,没有也不加载,很少使用
- timeoutInterval:超时时长,默认60s
另外,还可以设置其它一些信息,比如请求头,请求体等等,如下:
注意,下面的request应为NSMutableURLRequest,即可变类型
// 告诉服务器数据为json类型
[request setValue:@"application/json" forHTTPHeaderField:@"Content-Type"];
// 设置请求体(json类型)
NSData *jsonData = [NSJSONSerialization dataWithJSONObject:@{@"userid":@"123456"} options:NSJSONWritingPrettyPrinted error:nil];
request.HTTPBody = jsonData;
3 发送请求:
NSURLConnection默认的请求类型为GET,下面分异步和同步两种介绍
异步请求
[NSURLConnection sendAsynchronousRequest:request queue:[[NSOperationQueue alloc] init] completionHandler:^(NSURLResponse *response, NSData *data, NSError *connectionError) {
// 有的时候,服务器访问正常,但是会没有数据!
// 以下的 if 是比较标准的错误 处理代码!
if (connectionError != nil || data == nil) {
//给用户的提示信息
NSLog(@"网络不给力");
return;
}
}];
参数说明如下:
- completionHandler:请求响应后(或者请求超时)执行的代码,queue为代码添加到的队列,即block执行的线程
- NSURLResponse 为服务器的响应,真实类型为NSHTTPURLResponse,通常只在“下载”功能时,才会使用;下面是协议头的参数:
- URL:响应的URL,有的时候,访问一个URL地址,服务器可能会出现重定向,会定位到新的地址!
- MIMEType(Content-Type):服务器告诉客户端,可以用什么软件打开二进制数据!网络之所以丰富多采,是因为有丰富的客户端软件!栗子:windows上提示安装 Flash 插件
- expectedContentLength:预期的内容长度,要下载的文件长度,下载文件时非常有用
- suggestedFilename:"建议"的文件名,方便用户直接保存,很多时候,用户并不关心要保存成什么名字!
- textEncodingName:文本的编码名称 @"UTF8",大多数都是 UTF8
- statusCode:状态码,在做下载操作的时候,需要判断一下
- allHeaderFields:所有的响应头字典时候,用户并不关心要保存成什么名字!
- NSData 服务器返回的数据,例如json、xml(现在用的少)
- NSError 网络访问错误码
- NSURLResponse 为服务器的响应,真实类型为NSHTTPURLResponse,通常只在“下载”功能时,才会使用;下面是协议头的参数:
注意,block的执行线程为queue,如果block涉及到UI操作,则必须回到主线程:[NSOperationQueue mainQueue]
发送同步请求
// 同步请求,代码会阻塞在这里一直等待服务器返回,如果data为nil则请求失败,当获取少量数据时可以使用此方法。
// request参数同上。
NSData *data = [NSURLConnection sendSynchronousRequest:request returningResponse:nil error:nil];
三、举例说明
分三个部分:
- 网络请求(json、xml数据)
- 文件下载
- 文件上传,这里例子不再给出,请参考这里:http://www.cnblogs.com/mddblog/p/5215453.html
1 一般的网络请求
/// 两种请求方式
typedef enum {
MethodGET,
MethodPOST
}Method;
/// 请求成功后,直接将jsonData转为jsonDict
+ (void)connectionRequestWithMethod:(Method)method URLString:(NSString *)URLString parameters:(NSDictionary *)dict success:(void (^)(id JSON))success fail:(void (^)(NSError *error))fail {
// 简单的转码,如果参数带有?&特殊字符,下面方法不适合
URLString = [URLString stringByAddingPercentEscapesUsingEncoding:NSUTF8StringEncoding];
// 1.创建请求
NSURL *url = [NSURL URLWithString:URLString];
NSMutableURLRequest *request = [NSMutableURLRequest requestWithURL:url];
// 请求方式,默认为GET
if (method == MethodPOST) {
request.HTTPMethod = @"POST";
}
// 根据需要设置
[request setValue:@"text/html,application/xhtml+xml,application/xml;q=0.9,image/webp,*/*;q=0.8" forHTTPHeaderField:@"Accept"];
// 2.设置请求头 Content-Type 返回格式
[request setValue:@"application/json" forHTTPHeaderField:@"Content-Type"];
// 3.设置请求体 NSDictionary --> NSData
if (dict != nil) {
NSData *data = [NSJSONSerialization dataWithJSONObject:dict options:NSJSONWritingPrettyPrinted error:nil];
request.HTTPBody = data;
}
// 4.发送请求
[NSURLConnection sendAsynchronousRequest:request queue:[NSOperationQueue mainQueue] completionHandler:^(NSURLResponse *response, NSData *data, NSError *connectionError) {
if ((data != nil) && (connectionError == nil)) {
NSDictionary *jsonDict = [NSJSONSerialization JSONObjectWithData:data options:NSJSONReadingMutableContainers error:nil];
if (success) {
success(jsonDict);
}
} else {
if (fail) {
fail(connectionError);
}
}
}];
}
2 文件下载
下面举一个实现断点续传的网络下载,要实现断点续传,还需要用到NSURLConnectionDataDelegate代理,NSFileHandle文件操作或者数据流的形式(NSOutputStream),这里以NSFileHandle为例,访问的服务器采用Apache搭建的本地服务器,具体见代码:
@interface ViewController () <NSURLConnectionDataDelegate>
/// 文件下载完毕之后,在本地保存的路径
@property (nonatomic, copy) NSString *filePath;
/// 需要下载的文件的总大小!
@property (nonatomic, assign) long long serverFileLength;
/// 当前已经下载长度
@property (nonatomic, assign) long long localFileLength;
@end
@implementation ViewController
- (void)viewDidLoad {
[super viewDidLoad];
}
/// 点击屏幕事件
- (void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event {
[self downloadFile];
}
/// 下载文件
- (void)downloadFile {
// 文件存储路径
self.filePath = @"/Users/username/Desktop//陶喆 - 爱很简单.mp3";
// 要下载的网络文件,采用Apache搭建的本地服务器
NSString *urlStr = @"http://localhost/陶喆 - 爱很简单.mp3";
// 简单的转码,如果参数带有?&特殊字符,下面方法不适合
urlStr = [urlStr stringByAddingPercentEscapesUsingEncoding:NSUTF8StringEncoding];
NSURL *url = [NSURL URLWithString:urlStr];
NSMutableURLRequest *request = [NSMutableURLRequest requestWithURL:url];
// 设置请求方法为 HEAD 方法,这里只是头,数据量少,用同步请求也可以,不过最好还是用异步请求
request.HTTPMethod = @"HEAD";
// 无论是否会引起循环引用,block里面都使用弱引用
__weak typeof(self) weakSelf = self;
[NSURLConnection sendAsynchronousRequest:request queue:[[NSOperationQueue alloc] init] completionHandler:^(NSURLResponse * _Nullable response, NSData * _Nullable data, NSError * _Nullable connectionError) {
// 出错则返回
if (connectionError != nil) {
NSLog(@"connectionError = %@",connectionError);
return ;
}
// 记录需要下载的文件总长度
long long serverFileLength = response.expectedContentLength;
weakSelf.serverFileLength = serverFileLength;
// 初始化已下载文件大小
weakSelf.localFileLength = 0;
NSDictionary *dict = [[NSFileManager defaultManager] attributesOfItemAtPath:weakSelf.filePath error:NULL];
long long localFileLength = [dict[NSFileSize] longLongValue];
// 如果没有本地文件,直接下载!
if (!localFileLength) {
// 下载新文件
[weakSelf getFileWithUrlString:urlStr];
}
// 如果已下载的大小,大于服务器文件大小,肯定出错了,删除文件并从新下载
if (localFileLength > serverFileLength) {
// 删除文件 remove
[[NSFileManager defaultManager] removeItemAtPath:self.filePath error:NULL];
// 下载新文件
[self getFileWithUrlString:urlStr];
} else if (localFileLength == serverFileLength) {
NSLog(@"文件已经下载完毕");
} else if (localFileLength && localFileLength < serverFileLength) {
// 文件下载了一半,则使用断点续传
self.localFileLength = localFileLength;
[self getFileWithUrlString:urlStr WithStartSize:localFileLength endSize:serverFileLength-1];
}
}];
}
/// 重新开始下载文件(代理方法监听下载过程)
-(void)getFileWithUrlString:(NSString *)urlString
{
NSURL *url = [NSURL URLWithString:urlString];
// 默认就是 GET 请求
NSURLRequest *request = [NSURLRequest requestWithURL:url];
// 开始请求,并设置代理为self
NSURLConnection *conn = [[NSURLConnection alloc] initWithRequest:request delegate:self];
// 设置代理的执行线程,一般不在主线程
[conn setDelegateQueue:[[NSOperationQueue alloc] init]];
// NSUrlConnection 的代理方法是一个特殊的事件源!
// 开启运行循环!
CFRunLoopRun();
}
/* 断点续传(代理方法监听下载过程)
* startSize:本次断点续传开始的位置
* endSize:本地断点续传结束的位置
*/
-(void)getFileWithUrlString:(NSString *)urlString WithStartSize:(long long)startSize endSize:(long long)endSize
{
// 1. 创建请求!
NSURL *url = [NSURL URLWithString:urlString];
// 默认就是 GET 请求
NSMutableURLRequest *request = [NSMutableURLRequest requestWithURL:url];
// 设置断点续传信息
NSString *range = [NSString stringWithFormat:@"Bytes=%lld-%lld",startSize,endSize];
[request setValue:range forHTTPHeaderField:@"Range"];
// NSUrlConnection 下载过程!
NSURLConnection *conn = [[NSURLConnection alloc] initWithRequest:request delegate:self];
// 设置代理的执行线程// 传一个非主队列!
[conn setDelegateQueue:[[NSOperationQueue alloc] init]];
// NSUrlConnection 的代理方法是一个特殊的事件源!
// 开启运行循环!
CFRunLoopRun();
}
#pragma mark - NSURLConnectionDataDelegate
/// 1. 接收到服务器响应
- (void)connection:(NSURLConnection *)connection didReceiveResponse:(NSURLResponse *)response {
// 服务器响应
// response.expectedContentLength的大小是要下载文件的大小,而不是文件的总大小。比如请求头设置了Range[requestM setValue:rangeStr forHTTPHeaderField:@"Range"];则返回的大小为range范围内的大小
}
/// 2. 接收到数据
- (void)connection:(NSURLConnection *)connection didReceiveData:(NSData *)data {
//data当前接收到的网络数据;
// 如果这个文件不存在,响应的文件句柄就不会创建!
NSFileHandle *fileHandle = [NSFileHandle fileHandleForWritingAtPath:self.filePath];
// 在这个路径下已经有文件了!
if (fileHandle) {
// 将文件句柄移动到文件的末尾
[fileHandle seekToEndOfFile];
// 写入文件的意思(会将data写入到文件句柄所操纵的文件!)
[fileHandle writeData:data];
[fileHandle closeFile];
} else {
// 第一次调用这个方法的时候,在本地还没有文件路径(没有这个文件)!
[data writeToFile:self.filePath atomically:YES];
}
}
/// 3. 接收完成
- (void)connectionDidFinishLoading:(NSURLConnection *)connection {
NSLog(@"下载完成");
}
/// 4. 网络错误
- (void)connection:(NSURLConnection *)connection didFailWithError:(NSError *)error {
NSLog(@"下载错误 %@", error);
}
@end
四、NSData 数据的反序列化
服务器返回的NSData数据类型,事先已经知道,因此可以直接返序列化为实际的类型,例如:json(字典或数组)、.plist(字典或数组)、text、xml等:
- 加载数据
// 加载本地
NSString *path = [[NSBundle mainBundle] pathForResource:@"文件名 " ofType:nil];
NSData *jsonData = [NSData dataWithContentsOfFile:path];
// 从网络直接加载
NSURL *url = [[NSBundle mainBundle] URLForResource:@"topic_news.json" withExtension:nil];
NSData *data = [NSData dataWithContentsOfURL:url];
- .plist反序列化
这种很少使用,只是苹果自己的格式
// 关于选型参数
// NSPropertyListImmutable = 0,不可变
// NSPropertyListMutableContainers = 1 << 0,容器可变
// NSPropertyListMutableContainersAndLeaves = 1 << 1,容器和叶子可变
// 通常后续直接做字典转模型,不需要关心是否可变,所以一般直接赋值为0
id result = [NSPropertyListSerialization propertyListWithData:data options:0 format:NULL error:NULL];
- 关于json数据
- json数据的本质是字符串,根格式只有两种可能:数组或字典。
- 反序列化:从服务器接收到的二进制数据 转换成 字典或者数组
// 假设json数据位:[{键值对},{键值对}],则可以将数据转化为字典或数组
NSArray *dictArray = [NSJSONSerialization JSONObjectWithData:jsonData options:NSJSONReadingMutableContainers error:nil];- 序列化:将字典或者数组 转换成 二进制数据,准备发送给服务器
// obj为字典或数组
NSData *data = [NSJSONSerialization dataWithJSONObject:obj options:0 error:NULL];
// 转化前可以判断是否可以序列化:
+ (BOOL)isValidJSONObject:(id)obj; - 一个对象能够被转换成 JSON 必须符合以下条件:
- 顶级节点,必须是一个 NSArray or NSDictionary
- 所有的对象必须是 NSString, NSNumber, NSArray, NSDictionary, or NSNull
- 所有字典的 key 都必须是 NSString
- NSNumber 不能为空或者无穷大
- json数据的本质是字符串,根格式只有两种可能:数组或字典。
注意事项:
- 请求的缓存策略使用 NSURLRequestReloadIgnoringCacheData,忽略本地缓存
- 服务器响应结束后,要记录 Etag,服务器内容和本地缓存对比是否变化的重要依据!
- 在发送请求时,设置 If-None-Match,并且传入 etag
- 连接结束后,要判断响应头的状态码,如果是 304,说明本地缓存内容没有发生变化,此时可以使用本地缓存来加载数据,如果缓存文件被意外删除,程序依然运行但会报错,加载数据也失败:NSCachedURLResponse *cachedURLRes = [[NSURLCache sharedURLCache] cachedResponseForRequest:requestM];//cachedURLRes为nil
- GET 缓存的数据会保存在 Cache 目录中 /bundleId 下,Cache.db 中
iOS9访问http网页
- 在Info.plist中添加NSAppTransportSecurity类型Dictionary;
- 在NSAppTransportSecurity下添加NSAllowsArbitraryLoads类型Boolean,值设为YES
iOS NSURLConnection使用详解的更多相关文章
- iOS应用开发详解
<iOS应用开发详解> 基本信息 作者: 郭宏志 出版社:电子工业出版社 ISBN:9787121207075 上架时间:2013-6-28 出版日期:2013 年7月 开本:16开 ...
- 转载]IOS LBS功能详解[0](获取经纬度)[1](获取当前地理位置文本 )
原文地址:IOS LBS功能详解[0](获取经纬度)[1](获取当前地理位置文本作者:佐佐木小次郎 因为最近项目上要用有关LBS的功能.于是我便做一下预研. 一般说来LBS功能一般分为两块:一块是地理 ...
- iOS中-Qutarz2D详解及使用
在iOS中Qutarz2D 详解及使用 (一)初识 介绍 Quartz 2D是二维绘图引擎. 能完成的工作有: 绘制图形 : 线条\三角形\矩形\圆\弧等 绘制文字 绘制\生成图片(图像) 读取\生成 ...
- iOS 2D绘图详解(Quartz 2D)之路径(点,直线,虚线,曲线,圆弧,椭圆,矩形)
前言:一个路径可以包含由一个或者多个shape以及子路径subpath,quartz提供了很多方便的shape可以直接调用.例如:point,line,Arc(圆弧),Curves(曲线),Ellip ...
- iOS开发——Block详解
iOS开发--Block详解 1. Block是什么 代码块 匿名函数 闭包--能够读取其他函数内部变量的函数 函数变量 实现基于指针和函数指针 实现回调的机制 Block是一个非常有特色的语法,它可 ...
- iOS开发:详解Objective-C runTime
Objective-C总Runtime的那点事儿(一)消息机制 最近在找工作,Objective-C中的Runtime是经常被问到的一个问题,几乎是面试大公司必问的一个问题.当然还有一些其他问题也几乎 ...
- 了解iOS消息推送一文就够:史上最全iOS Push技术详解
本文作者:陈裕发, 腾讯系统测试工程师,由腾讯WeTest整理发表. 1.引言 开发iOS系统中的Push推送,通常有以下3种情况: 1)在线Push:比如QQ.微信等IM界面处于前台时,聊天消息和指 ...
- iOS开发者证书-详解
iOS开发者证书-详解/生成/使用 本文假设你已经有一些基本的Xcode开发经验, 并注册了iOS开发者账号. 相关基础 加密算法 现代密码学中, 主要有两种加密算法: 对称密钥加密 和 公开密钥加密 ...
- iOS开发-Runtime详解
iOS开发-Runtime详解 简介 Runtime 又叫运行时,是一套底层的 C 语言 API,其为 iOS 内部的核心之一,我们平时编写的 OC 代码,底层都是基于它来实现的.比如: [recei ...
随机推荐
- 揭开JDBC的神秘面纱,让JDBC数据库的连接参数不再神秘
1.JDBC是什么? JDBC(Java DataBase Connectivity)java数据库连接 2.JDBC可以做什么? 简单地说,JDBC 可做三件事:与数据库建立连接.发送 ...
- ListView中Button事件
为了解决ListView中Item里的Button独立事件响应,能够採用下面方法: 在BaseAdapter的getview里加入加粗代码: <span style="font-siz ...
- (三)Linux Shell编程——Shell常用命令(输出、判断、循环、函数、包含)
3. 常用命令 3.1 输出 3.1.1 echo命令 echo是Shell的一个内部指令,用于在屏幕上打印出指定的字符串.命令格式: echo arg name="coding" ...
- 主流ETL(Extract-Transform-Load)工具选型,Kettle Spoon、Datastage、Powercenter介绍
参考:三大主流ETL工具选型 ETL工具 Kettle Spoon 开源ETL工具,所以免费,用java开发的. Ascential公司的Datastage(在2005年被IBM收购现在是 IBM 的 ...
- Android应用程序结构总结
Android应用程序结构分析 由于是初学者,对于Android应用程序的结构的认识是一穷二白的,对于开发Android应用程序,必须先了解其程序的结构和作用.一下就用一个简单的例子来解剖: 从上图的 ...
- jquery动态绑定事件
什么是动态绑定? 动态绑定是指动态添加的DOM节点或者html元素,他们最开始时运行的时候是不存在的.如果要给这些动态加入的节点增加事件,就必须要用jquery的on方法来绑定事件. $('.cont ...
- CView类的使用
首先我们来写一个样例: 1.建一个win32简单应用程序,不要觉得这样就不能写出MFC程序,由于是不是MFC程序取决于调没调MFC函数. 2. 删除入口函数.仅仅留下#include "st ...
- Oracle免费版和付费版,各版本的说明
Oracle免费版和付费版的区别: 首先这里给出一个答案,oracle确实是免费的给学习的人员使用.oracle的策略就是你可以随意下载我的产品,包括oracle,weblogic等产品用于学习, ...
- notepad++与ISE/Vivado关联
转自:http://www.cnblogs.com/ninghechuan/p/6172237.html 1.notepad++与vivado关联 打开vivado软件,选择菜单栏“Tools——&g ...
- Spring里的FactoryBean和BeanFactory有啥区别?
分别看这俩文章就知道了 Spring的FactoryBean使用 Spring加载xml配置文件的方式 ApplicationContext