项目中经常会遇到一些重复性的劳动,别的不扯远,就说HTTP和SOCKET,发送请求,得到成功返回或失败返回。每次我都需要在不同的逻辑层次中处理成功、失败两个的回调,这样带来的劣势就是代码显得分散和冗长,重复代码很多。于是我利用block,将这一问题做了优化。本文利用ASIHttpRequest库,以最基本的HTTP请求,GET方法为例,说明这一方法。
最终的效果:
1 [[PPDHttpRequest shareInstance] requestWithUrlString:@"http://192.168.2.99:5009/message/init/1032/13/0/1" complection:^(id arg) { 2 // 3 NSLog(@"%@", arg); 4 } failure:^(id arg) { 5 // 6 NSLog(@"%@", arg); 7 }];
三个参数分别是Get方法的URL、成功block“回调”、失败block“回调”。请求与结果整合在同一函数,清晰明了,易于维护。
完整demo地址:https://github.com/pigpigdaddy/HttpRequestUsingBlockDemo
下面根据关键点来逐一阐述:
一:单例模式
采用单例模式也是为了简化、瘦身、统一管理。关于单例模式的使用方法、原理介绍等不在本文讨论。我认为一般在一个APP中,“底层”网络接口如HTTP接口,用一个类来做就OK了,所有HTTP网络请求都可以通过这一个类接口来进行。如果需要针对不同模块功能分开调用,则可以在此类上在做封装,分出不同的类来负责不同模块的HTTP调用。那么既然底层只需要一个类且整个程序可能随时需要处理HTTP网络请求,那么我决定把他做成一个单例来处理。
1 @interface PPDHttpRequest : NSObject 2 3 /** 4 TODO:获取Instance单例 5 6 @return URHTTPRequest 实例对象 7 8 @author pigpigdaddy 9 @since 10 */ 11 + (instancetype)shareInstance; 12 13 @end
1 /** 2 TODO:获取Instance单例 3 4 @return URHTTPRequest 实例对象 5 6 @author pigpigdaddy 7 @since 8 */ 9 + (instancetype)shareInstance{ 10 static PPDHttpRequest *httpRequest; 11 static dispatch_once_t onceToken; 12 dispatch_once(&onceToken,^{ 13 httpRequest = [[PPDHttpRequest alloc] init]; 14 }); 15 return httpRequest; 16 }
以上是标准的ARC环境下创建单例的代码。
二,添加请求接口,方法参数包含成功、失败block
调用方法是本文的关键采用了如下方式:
1 /** 2 TODO:http请求 3 4 @param urlString 请求地址 5 @param parameters 请求参数 6 @param complection 完成block 7 @param failure 失败block 8 9 @author pigpigdaddy 10 @since 3.0 11 */ 12 - (void)requestWithUrlString:(NSString *)urlString complection:(commonBlock)complection failure:(commonBlock)failure;
1 /** 2 TODO:http请求 3 4 @param urlString 请求地址 5 @param parameters 请求参数 6 @param complection 完成block 7 @param failure 失败block 8 9 @author pigpigdaddy 10 @since 3.0 11 */ 12 - (void)requestWithUrlString:(NSString *)urlString complection:(commonBlock)complection failure:(commonBlock)failure 13 { 14 NSLog(@"---------------PPDHttpRequest url:%@-----------",urlString); 15 NSURL *url=[NSURL URLWithString:urlString]; 16 ASIHTTPRequest *request = [[ASIHTTPRequest alloc] initWithURL:url]; 17 [ASIHTTPRequest setDefaultTimeOutSeconds:60]; 18 [request setDelegate:self]; 19 [request setRequestMethod:@"GET"]; 20 [request setShouldAttemptPersistentConnection:NO]; 21 22 //添加信息对象 23 NSMutableDictionary *userInfo = [NSMutableDictionary dictionary]; 24 if (complection){ 25 [userInfo setObject:complection forKey:@"complectionBlock"]; 26 } 27 if (failure){ 28 [userInfo setObject:failure forKey:@"failureBlock"]; 29 } 30 [request setUserInfo:userInfo]; 31 32 [self.queue addOperation:request]; 33 }
可以看到,接口调用除了Get方法URL以外,还包含了两个block,这两个block的类型我通过以下方式定义:
1 //定义通用的block 2 typedef void(^commonBlock)(id arg);
拥有一个id类型的参数,用于“返回值”。
在方法实现中,我将两个block设置到userInfo中,这样,可以在http回调中,进行相应的处理。
三,处理回调
正如第二条所指出的那样,在http回调中,让相应的block做出正确响应:
1 - (void)requestFinished:(ASIHTTPRequest *)request{ 2 NSDictionary *userInfo = request.userInfo; 3 commonBlock complection = [userInfo objectForKey:@"complectionBlock"]; 4 if (complection){ 5 complection(request.responseString); 6 } 7 } 8 9 - (void)requestFailed:(ASIHTTPRequest *)request{ 10 NSDictionary *userInfo = request.userInfo; 11 commonBlock failure = [userInfo objectForKey:@"failureBlock"]; 12 if (failure){ 13 failure(request.responseString); 14 } 15 }
最终的效果就如文章开头贴出的那段代码一样,将请求、回调整合在了一起。代码简洁明了,易读性高。也易于二次封装。
多说一句,上层封装时,可以开出不同的方法,接受各种参数,拼接成URL(get),或是表单(post),以及最重要的是,传入成功失败block参数,即可!
完整demo地址:https://github.com/pigpigdaddy/HttpRequestUsingBlockDemo
参考文档:
http://www.cnblogs.com/eagle927183/p/3457538.html
http://sjpsega.com/blog/2014/05/25/singleton-in-ios/
http://blog.csdn.net/gideal_wang/article/details/4316691