在IOS开发过程中,ASIHttpRequest库是最常用的网络库,功能强大,使用也非常方便。
但是,在使用此库过程中,发现有几点小问题。网络上仔细搜索研究了好久,现记录于此。
一个问题是,我发现当异步请求比较多,并发连接数量比较多的时候,会导致一些请求失败。
原因:
代码默认是最大4个并发连接,其他的连接需要等待。然后如果有连接请求完毕了,就会去复用这个连接,但是还会出现连接关闭的情况,这个时候代码不会再一次重新请求,导致这次请求失败,返回nil。
简单解决:
最简单的解决就是把复用连接关闭,每次都新建连接就不会出问题。或者一次只允许一个连接。
ASIHttpRequest里加入下面代码 [self setShouldAttemptPersistentConnection:NO];
另一个问题是偶尔出现异常,在ASIHTTPRequest的startAsynchronous调用后偶尔出现EXC_BAD_ACCESS异常。
在ViewController中是采用委托的方式,即在- (void)viewDidLoad方法中加入代码:[request setDelegate:self]
然后实现其委托方法- (void)requestFinished:(ASIHTTPRequest *)request和- (void)requestFailed:(ASIHTTPRequest *)request
测试过程中发现,在执行请求未结束时候不断关闭和打开此界面,就容易出现EXC_BAD_ACCESS异常。
解决方法:
在stackoverflow.com上,有网友提出了两个解决方法。
第一个,使用同一的delegate,并且应用一运行就初始化它。如果其他类要调用异步请求,将delegate发过去。
调用方法如下:
- (void)sendUrl: (NSString *) restUrl withCallBack:(NSObject*) delegate;
然后,在delegate的实现方法中利用request的tag来标识来自哪一个类的异步调用。
第二个,使用block直接实现异步请求,抛弃delegate。
使用"setCompletionBlock:^"代替delegate的"setDidFinishSelector"。
request做一个设置就可以,比原来通过设置request.delegate再实现delegate的方法简单多了。
[request setCompletionBlock :^{
// 请求响应结束,返回 responseString
NSString *responseString = [request responseString ]; // 对于 2 进制数据,使用 NSData 返回 NSData *responseData = [request responseData];
NSLog ( @"%@" ,responseString);
}];
[request setFailedBlock :^{
// 请求响应失败,返回错误信息
NSError *error = [request error ];
NSLog ( @"error:%@" ,[error userInfo ]);
}];
确实管用