iOS开发网络篇—NSURLConnection基本使用(一)

本文涉及的产品
全局流量管理 GTM,标准版 1个月
云解析 DNS,旗舰版 1个月
公共DNS(含HTTPDNS解析),每月1000万次HTTP解析
简介:

一、NSURLConnection的常用类

(1)NSURL:请求地址

(2)NSURLRequest:封装一个请求,保存发给服务器的全部数据,包括一个NSURL对象,请求方法、请求头、请求体....

(3)NSMutableURLRequest:NSURLRequest的子类

(4)NSURLConnection:负责发送请求,建立客户端和服务器的连接。发送NSURLRequest的数据给服务器,并收集来自服务器的响应数据

补充:一个NSURLRequest对象就代表一个请求,它包含的信息有:

    一个NSURL对象

    请求方法、请求头、请求体

    请求超时

    …...

二、NSURLConnection的使用

1.简单说明

使用NSURLConnection发送请求的步骤很简单

(1)创建一个NSURL对象,设置请求路径(设置请求路径)

(2)传入NSURL创建一个NSURLRequest对象,设置请求头和请求体(创建请求对象)

(3)使用NSURLConnection发送NSURLRequest(发送请求)

2.代码示例

(1)发送请求的三个步骤:

1.设置请求路径

2.创建请求对象

3.发送请求

3.1发送同步请求(一直在等待服务器返回数据,这行代码会卡住,如果服务器,没有返回数据,那么在主线程UI会卡住不能继续执行操作)有返回值

3.2发送异步请求:没有返回值

说明:任何NSURLRequest默认都是get请求。

(2)发送同步请求代码示例:

复制代码 复制代码
 1 //
 2 // YYViewController.m
 3 // 01-NSURLConnection的使用(GET)  4 //  5 // Created by apple on 14-6-28.  6 // Copyright (c) 2014年 itcase. All rights reserved.  7 //  8 9 #import "YYViewController.h" 10 #import "MBProgressHUD+MJ.h" 11 12 @interface YYViewController () 13 @property (weak, nonatomic) IBOutlet UITextField *username; 14 @property (weak, nonatomic) IBOutlet UITextField *pwd; 15 - (IBAction)login; 16 17 @end 18 19 @implementation YYViewController 20 21 - (IBAction)login { 22 // 1.提前的表单验证 23 if (self.username.text.length==0) { 24 [MBProgressHUD showError:@"请输入用户名"]; 25 return; 26  } 27 if (self.pwd.text.length==0) { 28 [MBProgressHUD showError:@"请输入密码"]; 29 return; 30  } 31 // 2.发送请求给服务器(带上账号和密码) 32 //添加一个遮罩,禁止用户操作 33 // [MBProgressHUD showMessage:@"正在努力加载中...."]; 34 // GET请求:请求行\请求头\请求体 35 // 36 // 1.设置请求路径 37 NSString *urlStr=[NSString stringWithFormat:@"http://192.168.1.53:8080/MJServer/login?username=%@&pwd=%@",self.username.text,self.pwd.text]; 38 NSURL *url=[NSURL URLWithString:urlStr]; 39 // 2.创建请求对象 40 NSURLRequest *request=[NSURLRequest requestWithURL:url]; 41 // 3.发送请求 42 //发送同步请求,在主线程执行 43 NSData *data=[NSURLConnection sendSynchronousRequest:request returningResponse:nil error:nil]; 44 //(一直在等待服务器返回数据,这行代码会卡住,如果服务器没有返回数据,那么在主线程UI会卡住不能继续执行操作) 45 NSLog(@"--%d--",data.length); 46 } 47 @end
复制代码 复制代码

模拟器情况:

打印服务器返回的信息:

补充说明:

1.提前的表单验证

2.发送请求给服务器(带上账号和密码)

GET请求:请求行\请求头\请求体

注意:GET请求中不存在请求体,因为所有的信息都写在URL里面。在IOS里面,请求行和请求头都不用写。

(3)发送异步请求

发送异步请求有两种方式:

1)使用block回调

2)代理

A.使用block回调方法发送异步请求

使用block回调代码示例:

复制代码 复制代码
 1 //
 2 // YYViewController.m
 3 // 01-NSURLConnection的使用(GET)  4 //  5 // Created by apple on 14-6-28.  6 // Copyright (c) 2014年 itcase. All rights reserved.  7 //  8 9 #import "YYViewController.h" 10 #import "MBProgressHUD+MJ.h" 11 12 @interface YYViewController () 13 @property (weak, nonatomic) IBOutlet UITextField *username; 14 @property (weak, nonatomic) IBOutlet UITextField *pwd; 15 - (IBAction)login; 16 17 @end 18 19 @implementation YYViewController 20 21 - (IBAction)login { 22 // 1.提前的表单验证 23 if (self.username.text.length==0) { 24 [MBProgressHUD showError:@"请输入用户名"]; 25 return; 26  } 27 if (self.pwd.text.length==0) { 28 [MBProgressHUD showError:@"请输入密码"]; 29 return; 30  } 31 // 2.发送请求给服务器(带上账号和密码) 32 //添加一个遮罩,禁止用户操作 33 [MBProgressHUD showMessage:@"正在努力加载中...."]; 34 35 // 36 // 1.设置请求路径 37 NSString *urlStr=[NSString stringWithFormat:@"http://192.168.1.53:8080/MJServer/login?username=%@&pwd=%@",self.username.text,self.pwd.text]; 38 NSURL *url=[NSURL URLWithString:urlStr]; 39 40 // 2.创建请求对象 41 NSURLRequest *request=[NSURLRequest requestWithURL:url]; 42 43 // 3.发送请求 44 //3.1发送同步请求,在主线程执行 45 // NSData *data=[NSURLConnection sendSynchronousRequest:request returningResponse:nil error:nil]; 46 //(一直在等待服务器返回数据,这行代码会卡住,如果服务器没有返回数据,那么在主线程UI会卡住不能继续执行操作) 47 48 //3.1发送异步请求 49 //创建一个队列(默认添加到该队列中的任务异步执行) 50 // NSOperationQueue *queue=[[NSOperationQueue alloc]init]; 51 //获取一个主队列 52 NSOperationQueue *queue=[NSOperationQueue mainQueue]; 53 [NSURLConnection sendAsynchronousRequest:request queue:queue completionHandler:^(NSURLResponse *response, NSData *data, NSError *connectionError) { 54 NSLog(@"--block回调数据--%@---%d", [NSThread currentThread],data.length); 55 //隐藏HUD,刷新UI的操作一定要放在主线程执行 56  [MBProgressHUD hideHUD]; 57 58 //解析data 59 /* 60  {"success":"登录成功"} 61  {"error":"用户名不存在"} 62  {"error":"密码不正确"} 63 */ 64 NSDictionary *dict=[NSJSONSerialization JSONObjectWithData:data options:NSJSONReadingMutableLeaves error:nil]; 65 NSLog(@"%@",dict); 66 67 //判断后,在界面提示登录信息 68 NSString *error=dict[@"error"]; 69 if (error) { 70  [MBProgressHUD showError:error]; 71 }else 72  { 73 NSString *success=dict[@"success"]; 74 [MBProgressHUD showSuccess:success]; 75 } 76 }]; 77 NSLog(@"请求发送完毕"); 78 } 79 @end
复制代码 复制代码

模拟器情况(注意这里使用了第三方框架):

打印查看:

代码说明:

block代码段:当服务器有返回数据的时候调用会开一条新的线程去发送请求,主线程继续往下走,当拿到服务器的返回数据的数据的时候再回调block,执行block代码段。这种情况不会卡住主线程。

队列的作用:决定这个block操作放在哪个线程执行?

刷新UI界面的操作应该放在主线程执行,不能放在子线程,在子线程处理UI相关操作会出现一些莫名的问题。

提示:

(1)创建一个操作,放在NSOperation队列中执行,默认是异步执行的。

(2)mainqueue 返回一个和主线程相关的队列,即主队列。

新的问题:如果向服务器发送请求,却并没有拿到数据,那么程序会崩溃(data不能为空)

改进代码:

复制代码 复制代码
 1 NSOperationQueue *queue=[NSOperationQueue mainQueue];
 2 [NSURLConnection sendAsynchronousRequest:request queue:queue completionHandler:^(NSURLResponse *response, NSData *data, NSError *connectionError) {
 3 //当请求结束的时候调用(有两种结果,一个是成功拿到数据,也可能没有拿到数据,请求失败)  4 NSLog(@"--block回调数据--%@---%d", [NSThread currentThread],data.length);  5 //隐藏HUD,刷新UI的操作一定要放在主线程执行  6  [MBProgressHUD hideHUD];  7 8 //解析data  9 /* 10  {"success":"登录成功"} 11  {"error":"用户名不存在"} 12  {"error":"密码不正确"} 13 */ 14 if (data) {//请求成功 15 NSDictionary *dict=[NSJSONSerialization JSONObjectWithData:data options:NSJSONReadingMutableLeaves error:nil]; 16 NSLog(@"%@",dict); 17 18 //判断后,在界面提示登录信息 19 NSString *error=dict[@"error"]; 20 if (error) { 21  [MBProgressHUD showError:error]; 22 }else 23  { 24 NSString *success=dict[@"success"]; 25  [MBProgressHUD showSuccess:success]; 26  } 27 }else //请求失败 28  { 29 [MBProgressHUD showError:@"网络繁忙,请稍后重试!"]; 30  } 31 32 }];
复制代码 复制代码

解析data

 //解析data
 /*
 {"success":"登录成功"}
 {"error":"用户名不存在"}
 {"error":"密码不正确"}
 */

说明:使用NSJSONSerialization 返回的对象,取决于最外层是什么,如果是{}那就是字典,[]那就是数组等。

补充说明:

首先确定请求路径,然后创建请求对象(默认发送的时get请求),使用异步方法(一调 用这个方法,它会自动开启一个子线程去发送请求,当请求成功,数据返回的时候自动调用内部的代码段,这个代码段在那个线程执行取决于队列,如果是主队列, 那么在子线程发送请求成功拿到服务器的数据后,回到主线程中解析数据,刷新UI界面)。

B.使用代理方法发送异步请求

要监听服务器返回的data,所以使用<NSURLConnectionDataDelegate>协议

常见大代理方法如下:

复制代码 复制代码
 1 #pragma mark- NSURLConnectionDataDelegate代理方法
 2 
 3 //当接收到服务器的响应(连通了服务器)时会调用
 4 
 5 -(void)connection:(NSURLConnection *)connection didReceiveResponse:(NSURLResponse *)response  6 7 //当接收到服务器的数据时会调用(可能会被调用多次,每次只传递部分数据)  8 9 -(void)connection:(NSURLConnection *)connection didReceiveData:(NSData *)data 10 11 //当服务器的数据加载完毕时就会调用 12 13 -(void)connectionDidFinishLoading:(NSURLConnection *)connection 14 15 //请求错误(失败)的时候调用(请求超时\断网\没有网\,一般指客户端错误) 16 17 -(void)connection:(NSURLConnection *)connection didFailWithError:(NSError *)error 
复制代码 复制代码

使用异步方法发送get请求的代码示例:

复制代码 复制代码
 1 //
 2 // YYViewController.m
 3 // 01-NSURLConnection的使用(GET)  4 //  5 // Created by apple on 14-6-28.  6 // Copyright (c) 2014年 itcase. All rights reserved.  7 //  8 9 #import "YYViewController.h"  10 #import "MBProgressHUD+MJ.h"  11 12 @interface YYViewController ()<NSURLConnectionDataDelegate>  13 @property (weak, nonatomic) IBOutlet UITextField *username;  14 @property (weak, nonatomic) IBOutlet UITextField *pwd;  15 @property(nonatomic,strong)NSMutableData *responseData;  16 - (IBAction)login;  17 18 @end  19 20 @implementation YYViewController  21 22 - (IBAction)login {  23 // 1.提前的表单验证  24 if (self.username.text.length==0) {  25 [MBProgressHUD showError:@"请输入用户名"];  26 return;  27  }  28 if (self.pwd.text.length==0) {  29 [MBProgressHUD showError:@"请输入密码"];  30 return;  31  }  32 // 2.发送请求给服务器(带上账号和密码)  33 //添加一个遮罩,禁止用户操作  34 [MBProgressHUD showMessage:@"正在努力加载中...."];  35 36 //  37 // 2.1设置请求路径  38 NSString *urlStr=[NSString stringWithFormat:@"http://192.168.1.53:8080/MJServer/login?username=%@&pwd=%@",self.username.text,self.pwd.text];  39 NSURL *url=[NSURL URLWithString:urlStr];  40 41 // 2.2创建请求对象  42 // NSURLRequest *request=[NSURLRequest requestWithURL:url];//默认就是GET请求  43 //设置请求超时  44 NSMutableURLRequest *request=[NSMutableURLRequest requestWithURL:url];  45 request.timeoutInterval=5.0;  46 47 // 2.3.发送请求  48 //使用代理发送异步请求(通常应用于文件下载)  49 NSURLConnection *conn=[NSURLConnection connectionWithRequest:request delegate:self];  50  [conn start];  51 NSLog(@"已经发出请求---");  52 }  53 54 #pragma mark- NSURLConnectionDataDelegate代理方法  55 /*  56  *当接收到服务器的响应(连通了服务器)时会调用  57 */  58 -(void)connection:(NSURLConnection *)connection didReceiveResponse:(NSURLResponse *)response  59 {  60 NSLog(@"接收到服务器的响应");  61 //初始化数据  62 self.responseData=[NSMutableData data];  63 }  64 65 /*  66 *当接收到服务器的数据时会调用(可能会被调用多次,每次只传递部分数据)  67 */  68 -(void)connection:(NSURLConnection *)connection didReceiveData:(NSData *)data  69 {  70 NSLog(@"接收到服务器的数据");  71 //拼接数据  72  [self.responseData appendData:data];  73 NSLog(@"%d---%@--",self.responseData.length,[NSThread currentThread]);  74 } 75 76 /* 77 *当服务器的数据加载完毕时就会调用 78 */ 79 -(void)connectionDidFinishLoading:(NSURLConnection *)connection 80 { 81 NSLog(@"服务器的数据加载完毕"); 82 //隐藏HUD 83 [MBProgressHUD hideHUD]; 84 85 //处理服务器返回的所有数据 86 NSDictionary *dict=[NSJSONSerialization JSONObjectWithData:self.responseData options:NSJSONReadingMutableLeaves error:nil]; 87 88 //判断后,在界面提示登录信息 89 NSString *error=dict[@"error"]; 90 if (error) { 91 [MBProgressHUD showError:error]; 92 }else 93 { 94 NSString *success=dict[@"success"]; 95 [MBProgressHUD showSuccess:success]; 96 } 97 NSLog(@"%d---%@--",self.responseData.length,[NSThread currentThread]); 98 } 99 /* 100 *请求错误(失败)的时候调用(请求超时\断网\没有网\,一般指客户端错误) 101 */ 102 -(void)connection:(NSURLConnection *)connection didFailWithError:(NSError *)error 103 { 104 // NSLog(@"请求错误"); 105 //隐藏HUD 106 [MBProgressHUD hideHUD]; 107 [MBProgressHUD showError:@"网络繁忙,请稍后重试!"]; 108 } 109 @end
复制代码 复制代码

打印查看:

补充:

(1)数据的处理

在didReceiveData:方法中,拼接接收到的所有数据,等所有数据都拿到后,在connectionDidFinishLoading:方法中进行处理

(2)网络延迟

在做网络开发的时候,一定要考虑到网络延迟情况的处理,可以在服务器的代码设置一个断点模拟。

在服务器代码的登录方法中设置断点

设置请求的最大延迟

模拟器情况:

打印查看:

三、NSMutableURLRequest

NSMutableURLRequest是NSURLRequest的子类,常用方法有

设置请求超时等待时间(超过这个时间就算超时,请求失败)- (void)setTimeoutInterval:(NSTimeInterval)seconds;

设置请求方法(比如GET和POST)- (void)setHTTPMethod:(NSString *)method;

设置请求体- (void)setHTTPBody:(NSData *)data;

设置请求头- (void)setValue:(NSString *)value forHTTPHeaderField:(NSString *)field;


相关文章
|
24天前
|
安全 网络安全 Android开发
安卓与iOS开发:选择的艺术网络安全与信息安全:漏洞、加密与意识的交织
【8月更文挑战第20天】在数字时代,安卓和iOS两大平台如同两座巍峨的山峰,分别占据着移动互联网的半壁江山。它们各自拥有独特的魅力和优势,吸引着无数开发者投身其中。本文将探讨这两个平台的特点、优势以及它们在移动应用开发中的地位,帮助读者更好地理解这两个平台的差异,并为那些正在面临选择的开发者提供一些启示。
114 56
|
16天前
|
C++
C++ Qt开发:QUdpSocket网络通信组件
QUdpSocket是Qt网络编程中一个非常有用的组件,它提供了在UDP协议下进行数据发送和接收的能力。通过简单的方法和信号,可以轻松实现基于UDP的网络通信。不过,需要注意的是,UDP协议本身不保证数据的可靠传输,因此在使用QUdpSocket时,可能需要在应用层实现一些机制来保证数据的完整性和顺序,或者选择在适用的场景下使用UDP协议。
53 2
|
16天前
|
小程序 数据安全/隐私保护
Taro@3.x+Vue@3.x+TS开发微信小程序,网络请求封装
在 `src/http` 目录下创建 `request.ts` 文件,并配置 Taro 的网络请求方法 `Taro.request`,支持多种 HTTP 方法并处理数据加密。
Taro@3.x+Vue@3.x+TS开发微信小程序,网络请求封装
|
24天前
|
数据采集 存储 前端开发
豆瓣评分9.0!Python3网络爬虫开发实战,堪称教学典范!
今天我们所处的时代是信息化时代,是数据驱动的人工智能时代。在人工智能、物联网时代,万物互联和物理世界的全面数字化使得人工智能可以基于这些数据产生优质的决策,从而对人类的生产生活产生巨大价值。 在这个以数据驱动为特征的时代,数据是最基础的。数据既可以通过研发产品获得,也可以通过爬虫采集公开数据获得,因此爬虫技术在这个快速发展的时代就显得尤为重要,高端爬虫人才的收人也在逐年提高。
|
15天前
|
安全 网络安全 数据安全/隐私保护
网络安全与信息安全:关于网络安全漏洞、加密技术、安全意识等方面的知识分享安卓与iOS开发中的线程管理比较
【8月更文挑战第30天】本文将探讨网络安全与信息安全的重要性,并分享关于网络安全漏洞、加密技术和安全意识的知识。我们将了解常见的网络攻击类型和防御策略,以及如何通过加密技术和提高安全意识来保护个人和组织的信息安全。
|
16天前
|
安全 网络安全 Android开发
探索安卓开发之旅:从新手到专家网络安全与信息安全:防范网络威胁,保护数据安全
【8月更文挑战第29天】在这篇技术性文章中,我们将踏上一段激动人心的旅程,探索安卓开发的世界。无论你是刚开始接触编程的新手,还是希望提升技能的资深开发者,这篇文章都将为你提供宝贵的知识和指导。我们将从基础概念入手,逐步深入到安卓开发的高级主题,包括UI设计、数据存储、网络通信等方面。通过阅读本文,你将获得一个全面的安卓开发知识体系,并学会如何将这些知识应用到实际项目中。让我们一起开启这段探索之旅吧!
|
物联网 Android开发 iOS开发
iOS开发 - 蓝牙学习的总结
iOS开发 - 蓝牙学习的总结
167 0
|
iOS开发
IOS开发---菜鸟学习之路--(十二)-利用ASIHTTPRequest进行异步获取数据
想要实现异步获取的话我这边了解过来有两个非常简单的方式 一个是利用ASIHTTPRequest来实现异步获取数据 另一个则是利用MBProgressHUD来实现异步获取数据 本章就先来讲解如何利用ASIHTTPRequest类来实现异步数据获取 首先大家需要百度一下ASIHTTPRequest 然后看一下百度里搜到的那些文章(不要问具体是那篇,因为我发现百度搜过来的东西全部都是一样的,所以。
1001 0
|
iOS开发
IOS开发---菜鸟学习之路--(九)-利用PullingRefreshTableView实现下拉刷新
本章主要讲解如何利用PullingRefreshTableView实现下拉(上拉)刷新的操作  PullingRefreshTableView 实现上下拉刷新的例子百度有很多,大家可以自己搜索下,先看下那些例子(一般搜索过来的都是一样的大家反正先把那部分内容先了解一下,然后再看本文档比较好。
876 0
|
iOS开发 Android开发 存储
IOS开发---菜鸟学习之路--(十)-实现新闻详细信息浏览页面
前面已经将了上下拉刷新 实现了上下拉刷新后我们的第一级界面就做好,接下来我们就需要实现 新闻详细信息浏览了 我个人认为一般实现新闻详细页面的方法有两种(主要是数据源的不同导致了方法的不同) 第一种是本身新闻就是一个链接地址,同时是已经处理好的适应手机浏览的网页 对于这种类型的数据源,我们直接在页面中放一个WebView控件,然后将URL传递过去就好了 另一种则是普通的包含标题、时间、内容、图片等数据结构的新闻内容(我们要实现的也是这种新闻,因为实现了这种之后, 我们就可以实现任何自定义的详细信息的页面了。
871 0