ios中使用asi崩溃的问题解决

隗翰海
2023-12-01

在IOS开发过程中,ASIHttpRequest库是最常用的网络库,功能强大,使用也非常方便。

但是,在使用此库过程中,发现有几点小问题。网络上仔细搜索研究了好久,现记录于此。

问题1:

一个问题是,我发现当异步请求比较多,并发连接数量比较多的时候,会导致一些请求失败。

原因:

代码默认是最大4个并发连接,其他的连接需要等待。然后如果有连接请求完毕了,就会去复用这个连接,但是还会出现连接关闭的情况,这个时候代码不会再一次重新请求,导致这次请求失败,返回nil。

简单解决:

最简单的解决就是把复用连接关闭,每次都新建连接就不会出问题。或者一次只允许一个连接。

ASIHttpRequest里加入下面代码 [self setShouldAttemptPersistentConnection:NO];

问题2:

另一个问题是偶尔出现异常,在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 ]);
    }];

确实管用

 类似资料: