以下内容为原创,欢迎转载,转载请注明

来自天天博客:http://www.cnblogs.com/tiantianbyconan/p/3409721.html

一般情况下的POST异步操作需要实现以下几步:

1. 在controller.h上实现<NSURLConnectionDataDelegate>协议

2. 实现协议的几个方法,

- (void)connection:(NSURLConnection *)connection didReceiveResponse:(NSURLResponse *)response

- (void)connection:(NSURLConnection *)connection didReceiveData:(NSData *)data

- (void)connectionDidFinishLoading:(NSURLConnection *)connection

3. 编写执行post请求的代码:

 NSURL *url = [NSURL URLWithString:urlStr]; // 生成NSURL对象
// 生成Request请求对象(并设置它的缓存协议、网络请求超时配置)
NSMutableURLRequest *request = [[NSMutableURLRequest alloc] initWithURL:url cachePolicy:NSURLRequestUseProtocolCachePolicy timeoutInterval:]; [request setHTTPBody:[params dataUsingEncoding:NSUTF8StringEncoding]]; // 设置请求参数 // 执行请求连接
NSURLConnection *conn = [[NSURLConnection alloc] initWithRequest:request delegate:executorDelegate];

如果controller有很多异步操作,处理就会很麻烦,而且,很多时候我们只需要处理完成和异常(比如超时)的时候的反馈即可

所以,我需要编写一个post请求的封装类,只要传入请求的url、请求参数(字符串形式)、完成时的回调block

首先,新建类:HttpPostExecutor,.h如下:

 //
// HttpPostExecutor.h
// HttpTest
//
// Created by WANGJIE on 13-11-6.
// Copyright (c) 2013年 WANGJIE. All rights reserved.
// #import <Foundation/Foundation.h> @interface HttpPostExecutor : NSObject<NSURLConnectionDataDelegate>
{
NSMutableData *resultData; // 存放请求结果
void (^finishCallbackBlock)(NSString *); // 执行完成后回调的block }
@property NSMutableData *resultData;
@property(strong) void (^finishCallbackBlock)(NSString *); + (void)postExecuteWithUrlStr:(NSString *)urlStr Paramters:(NSString *)params FinishCallbackBlock:(void (^)(NSString *))block; @end

实现了<NSURLConnectionDataDelegate>协议,因为它要接收post请求的几个回调。

有一个NSMutableData对象,这个对象用于储存请求的结果。

一个finishCallbackBlock的block,这个block用于执行完成后的回调,这个block传入的参数就是返回的结果(这个结果已转成utf-8编码的字符串形式),我们可以在这个block中去处理请求完成后的逻辑

还有一个类方法,这个类方法暴露给外面,让外面进行调用

接下来,我们看下实现的方法.m文件:

 //
// POST异步请求的封装
// 使用方法,只需传入url,参数组成的字符串,执行完成后的回调block
// 如下:
// [HttpPostExecutor postExecuteWithUrlStr:@"http://www.baidu.com"
// Paramters:@""
// FinishCallbackBlock:^(NSString *result){ // 设置执行完成的回调block
// NSLog(@"finish callback block, result: %@", result);
// }];
// post提交的参数,格式如下:
// 参数1名字=参数1数据&参数2名字=参数2数据&参数3名字=参数3数据&...
//
//
// HttpPostExecutor.m
// HttpTest
//
// Created by WANGJIE on 13-11-6.
// Copyright (c) 2013年 WANGJIE. All rights reserved.
// #import "HttpPostExecutor.h" @implementation HttpPostExecutor
@synthesize resultData, finishCallbackBlock; /**
* 执行POST请求
*/
+ (void)postExecuteWithUrlStr:(NSString *)urlStr Paramters:(NSString *)params FinishCallbackBlock:(void (^)(NSString *))block
{
// 生成一个post请求回调委托对象(实现了<NSURLConnectionDataDelegate>协议)
HttpPostExecutor *executorDelegate = [[HttpPostExecutor alloc] init];
executorDelegate.finishCallbackBlock = block; // 绑定执行完成时的block NSURL *url = [NSURL URLWithString:urlStr]; // 生成NSURL对象
// 生成Request请求对象(并设置它的缓存协议、网络请求超时配置)
NSMutableURLRequest *request = [[NSMutableURLRequest alloc] initWithURL:url cachePolicy:NSURLRequestUseProtocolCachePolicy timeoutInterval:]; [request setHTTPBody:[params dataUsingEncoding:NSUTF8StringEncoding]]; // 设置请求参数 // 执行请求连接
NSURLConnection *conn = [[NSURLConnection alloc] initWithRequest:request delegate:executorDelegate]; NSLog(conn ? @"连接创建成功" : @"连接创建失败"); } /**
* 接收到服务器回应的时回调
*/
- (void)connection:(NSURLConnection *)connection didReceiveResponse:(NSURLResponse *)response
{
NSHTTPURLResponse *resp = (NSHTTPURLResponse *)response;
// 初始化NSMutableData对象(用于保存执行结果)
if(!resultData){
resultData = [[NSMutableData alloc] init];
}else{
[resultData setLength:];
} if ([response respondsToSelector:@selector(allHeaderFields)]) { NSDictionary *dictionary = [resp allHeaderFields]; NSLog(@"[network]allHeaderFields:%@",[dictionary description]);
} }
/**
* 接收到服务器传输数据的时候调用,此方法根据数据大小执行若干次
*/
- (void)connection:(NSURLConnection *)connection didReceiveData:(NSData *)data
{
[resultData appendData:data]; // 追加结果
}
/**
* 数据传完之后调用此方法
*/
- (void)connectionDidFinishLoading:(NSURLConnection *)connection
{
// 把请求结果以UTF-8编码转换成字符串
NSString *resultStr = [[NSString alloc] initWithData:[self resultData] encoding:NSUTF8StringEncoding]; if (finishCallbackBlock) { // 如果设置了回调的block,直接调用
finishCallbackBlock(resultStr);
} }
/**
* 网络请求过程中,出现任何错误(断网,连接超时等)会进入此方法
*/
- (void)connection:(NSURLConnection *)connection didFailWithError:(NSError *)error
{
NSLog(@"network error: %@", [error localizedDescription]); if (finishCallbackBlock) { // 如果设置了回调的block,直接调用
finishCallbackBlock(nil);
} }
@end

在这个实现类中,我们在类方法中,先生成一个HttpPostExecutor对象,这个对象用于post请求的回调(因为实现了<NSURLConnectionDataDelegate>协议),然后去执行post连接。

接下来就等下面实现的回调方法被自动调用了,一旦调用

- (void)connection:(NSURLConnection *)connection didReceiveResponse:(NSURLResponse *)response

这个方法,就对resultData(用于存储post请求结果)进行初始化或者清空,因为要开始真正存储数据了嘛;

- (void)connection:(NSURLConnection *)connection didReceiveData:(NSData *)data

这个方法进行回调的时候,把返回过来的这部分数据存储到resultData中,没什么好说的;

一旦回调- (void)connectionDidFinishLoading:(NSURLConnection *)connection这个方法,说明数据传输完毕了,要做的逻辑就是把数据转成utf-8编码的字符串,然后回调我们设置的回调finishCallbackBlock,把转好的结果字符串传进去,这样我们在回调block方法中实现的逻辑就能正常执行了。

一旦回调- (void)connection:(NSURLConnection *)connection didFailWithError:(NSError *)error这个方法,说明请求过程中出错了,比如断电、超时等,这时候,也回调我们设置的回调finishCallbackBlock,nil作为结果,这样我们在finishCallbackBlock中就能判断是正常的执行了post还是出了问题。

好了,接下来,我们就可以在外面去调用了,如下:

 [HttpPostExecutor postExecuteWithUrlStr:@"http://www.baidu.com"
Paramters:@""
FinishCallbackBlock:^(NSString *result){
// 执行post请求完成后的逻辑
NSLog(@"finish callback block, result: %@", result);
}];

这样,以后post请求只需要去调用上面这个方法,在回调block中去处理结果

之后,在我们的代码编写中,就可以只关心业务逻辑,不需要去在意请求协议和回调了

测试demo下载

【ios】使用Block对POST异步操作的简单封装的更多相关文章

  1. iOS开发——UI篇OC篇&UITableView简单封装

    UITableView简单封装 UITableView时iOS开发中使用最多也是最重的一个UI空间,其实在App Store里面的%80以上的应用都用到了这个控件,所以就给大家介绍一下,前面的文章中也 ...

  2. iOS之断点下载,使用NSURLSession简单封装

    最近公司需要做个文件管理的功能模块,刚交到博主手上时,头都大了.因为没做过这方面的东西,只好咬牙加班,并请教某位大神,指点了一下,清楚研究方向,找了网上大量资料,最后实现简单的封装. 上代码:.h文件 ...

  3. iOS之block

    1. Block的声明和线程安全Block属性的声明,首先需要用copy修饰符,因为只有copy后的Block才会在堆中,栈中的Block的生命周期是和栈绑定的,可以参考之前的文章(iOS: 非ARC ...

  4. iOS中Block介绍(一)基础

    ios开发block的使用指南,以及深入理解block的内存管理,也适用于osx开发.讨论范围:block的使用,内存管理,内部实现.不包含的内容:gc arc下的block内存,block在c++中 ...

  5. iOS中Block介绍 基础

    ios开发block的使用指南,以及深入理解block的内存管理,也适用于osx开发.讨论范围:block的使用,内存管理,内部实现.不包含的内容:gc arc下的block内存,block在c++中 ...

  6. iOS 中Block以及Blocks的使用,闭包方法调用

    OC: -(void)dataWithUrl:(NSString*)string AndId:(NSInteger)id returnName:(void(^)(NSString*name))back ...

  7. iOS中block的用法 以及和函数用法的区别

    ios中block的用法和函数的用法大致相同 但是block的用法的灵活性更高: 不带参数的block: void ^(MyBlock)() = ^{}; 调用的时候  MyBlock(); 带参数的 ...

  8. iOS开发--Block

    iOS开发--Block 1.什么是Block,block 的作用 ui开发和网络常见功能实现回调,按钮的事件处理方法是回调方法以及网络下载后的回调处理 (1)按钮 target-action   一 ...

  9. iOS开发UI篇—UITableview控件简单介绍

    iOS开发UI篇—UITableview控件简单介绍 一.基本介绍 在众多移动应⽤用中,能看到各式各样的表格数据 . 在iOS中,要实现表格数据展示,最常用的做法就是使用UITableView,UIT ...

随机推荐

  1. Windows Azure Virtual Machine (31) 迁移Azure虚拟机

    <Windows Azure Platform 系列文章目录> 为什么要写这篇Blog? 之前遇到过很多客户提问: (1)我之前创建的虚拟机,没有加入虚拟网络.现在需要重新加入虚拟机网络, ...

  2. java中map插入相同的key

    测试用例: package test; import org.junit.Test; import po.Person; import java.util.HashMap; import java.u ...

  3. OpenJudge4980:拯救行动//stl优先队列

    总时间限制:  10000ms 内存限制:  65536kB 描述 公主被恶人抓走,被关押在牢房的某个地方.牢房用N*M (N, M <= 200)的矩阵来表示.矩阵中的每项可以代表道路(@). ...

  4. C# 解析html —— 将html转为XHTML,然后利用Xml解析

    呵呵,由于正则不熟,所以另谋出路——利用XML去解析html. 要想将抓取到的数据(直接抓取到的是byte[])  转为XML文档(即XMLDocument对象),有两个要点: 一.判断编码(http ...

  5. 实现多表关联来方便你的SELECT查询功能

    这次来学习一下多表关联.比如在数据中创建下面几张表,你可以从主键PRIMARY KEY与外键FOREIGN KEY来看到这几张表它们之间的关系. 场所表[dbo].[SixS_Location]: 主 ...

  6. Socket开发框架之消息的回调处理

    在一般的Socket应用里面,很多时候数据的发送和接收是分开处理的,也就是我们发送一个消息,不知道这个请求消息什么时候得到应答消息,而且收到对应的应答消息的时候,如果操作界面的内容,也是需要特别处理的 ...

  7. ASP.NET Core 开发 - Entity Framework (EF) Core

    EF Core 1.0 Database First http://www.cnblogs.com/linezero/p/EFCoreDBFirst.html ASP.NET Core 开发 - En ...

  8. 【C#】让DataGridView输入中实时更新数据源中的计算列

    本文适用Winform开发,且DataGridView的数据源为DataTable/DataView的情况. 理解前提:熟知DataTable.DataView 求:更好方案 考虑这样一个场景: 某D ...

  9. Asp.net Session 与Cookie的应用

    写程序的很多人都知道的,Session是服务器端的东西而Cooike是客户端的东西.因为B/S模式是无状态模式,它们的应用都是要存储客户端的某些登录或是加密的信息. Session存在于服务器端,当然 ...

  10. EXCEL countif函数多条件

    在MS EXCEL中,countif如果要满足多个条件怎么办呢? 1.答案就是:使用条件集合{}和sum函数. 即在countif()第二个参数中使用条件集合{},然后用sum()函数求满足这些条件的 ...