前几天优快云上有个介绍github上比较火的android开源项目,其中一个就是android-async-http,于是就下载下来学习了下。看了下源码,其实就是对apache httpclient的封装,可能httpclient使用起来比较繁琐,经这么封装以后,使用起来就很简单了,只需要在回调函数里写上自己想实现的代码即可。有一个功能我比较感兴趣,就是发生异常时的请求retry功能。以下代码就可以实现
httpClient.setHttpRequestRetryHandler(new RetryHandler(DEFAULT_MAX_RETRIES));
注意这里httpClient是DefaultHttpClient
其实httpclient里就有这个功能,这里还是进行了封装。项目中的RetryHandler类通过实现HttpRequestRetryHandler接口来实现retry功能。当异常发生时就会调用RetryHandler类中的public boolean retryRequest(IOException exception, int executionCount, HttpContext context)函数。但在这里有个异常黑名单,白名单,默认InterruptedIOException异常被放入到黑名单里去了,也就是说当这个异常发生后是不会进行retry的。InterruptedIOException有两个子类异常,ConnectTimeoutException, SocketTimeoutException,发生发生时也不会retry。但是我想要在发生这个两个异常时进行retry,于是将InterruptedIOException异常从异常黑名单中删掉。这时我以为当ConnectTimeoutException异常发生时就会触发retryRequest函数。但是当执行到
HttpUriRequest currentReq = (HttpUriRequest) context.getAttribute( ExecutionContext.HTTP_REQUEST );
String requestType = currentReq.getMethod();
就不再执行了,也不报错,retry功能也就实现不了了。而我在用httpclient实现的时候,当ConnectTimeoutException异常发生时,String requestType = currentReq.getMethod();是报错的,报空指针异常,应该是
HttpUriRequest currentReq = (HttpUriRequest) context.getAttribute( ExecutionContext.HTTP_REQUEST );
这句返回的是空,于是在String requestType = currentReq.getMethod();之前加上个判断currentReq是否为空就解决这个问题了。
if (currentReq != null) {
String requestType = currentReq.getMethod();retry = !requestType.equals("POST");
}
本文详细解析了Android-Async-HTTP项目中如何通过封装Apache HttpClient实现请求重试功能,特别关注了如何修改异常黑名单,使得ConnectTimeoutException和SocketTimeoutException异常也能触发重试操作。并通过实例代码演示了实现过程,对于提高应用的稳定性和用户体验具有重要意义。
531

被折叠的 条评论
为什么被折叠?



