Android OKHTTP 网络请求出错重连(结合Retrofit RxJava)--retryWhen命令符

本文主要是介绍Android OKHTTP 网络请求出错重连(结合Retrofit RxJava)--retryWhen命令符,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

链接:
https://www.jianshu.com/p/508c30aef0c1
网络抖动可能会产生大量的网络断链,所以httpClient自动重试是非常有必要的

需求1:根据异常类型选择是否重试
* 即,当发生的异常 = 网络异常 = IO异常 才选择重试
需求2:实现重试
* 通过返回的Observable发送的事件 = Next事件,从而使得retryWhen()重订阅,最终实现重试功能

需求3:延迟1段时间再重试
* 采用delay操作符 = 延迟一段时间发送,以实现重试间隔设置

需求4:遇到的异常越多,时间越长
* 在delay操作符的等待时间内设置 = 每重试1次,增多延迟重试时间0.5s
在这里插入图片描述

Retrofit.Builder builder = new Retrofit.Builder();builder.baseUrl(mBuilder.mBaseUrl).client(providerHttpClient()).addConverterFactory(GsonConverterFactory.create(gson)).addCallAdapterFactory(FRxJava2CallAdapterFactory.createWithScheduler());
public class FRxJava2CallAdapterFactory extends CallAdapter.Factory {private RxJava2CallAdapterFactory wrappedCallAdapterFactory;private FRxJava2CallAdapterFactory() {wrappedCallAdapterFactory = RxJava2CallAdapterFactory.createWithScheduler(Schedulers.io());}public static FRxJava2CallAdapterFactory createWithScheduler() {return new FRxJava2CallAdapterFactory();}@Overridepublic CallAdapter<?, ?> get(Type returnType, Annotation[] annotations, Retrofit retrofit) {CallAdapter<?, ?> callAdapter = wrappedCallAdapterFactory.get(returnType, annotations,retrofit);if (callAdapter == null) {return null;}Class<?> rawType = getRawType(returnType);boolean isCompletable = rawType == Completable.class;boolean isFlowable = rawType == Flowable.class;boolean isSingle = rawType == Single.class;boolean isMaybe = rawType == Maybe.class;return new FCallAdapter(callAdapter, isCompletable, isFlowable, isMaybe, isSingle);}private static class FCallAdapter<R> implements CallAdapter<R, Observable> {//we might handle other rawtype in the future. but currently only for Observable.private final boolean isCompletable;private final boolean isFlowable;private final boolean isMaybe;private final boolean isSingle;private CallAdapter wrapped;FCallAdapter(CallAdapter callAdapter, boolean isCompletable,boolean isFlowable, boolean isMaybe, boolean isSingle) {wrapped = callAdapter;this.isCompletable = isCompletable;this.isFlowable = isFlowable;this.isMaybe = isMaybe;this.isSingle = isSingle;}@Overridepublic Type responseType() {return wrapped.responseType();}@Overridepublic Observable<?> adapt(Call<R> call) {String retryOnFailure = call.request().header("retryOnFailure");boolean retry = retryOnFailure != null && Boolean.valueOf(retryOnFailure);return ((Observable) wrapped.adapt(call))
//                    .onErrorResumeNext()
//                    .timeout(10 * 1000, TimeUnit.MILLISECONDS).subscribeOn(Scheduler.ioThread()).observeOn(Scheduler.mainThread()).doAfterNext(new Consumer() {@Overridepublic void accept(Object o) throws Exception {
//                            if (!IOVCloudRepository.getInstance().getIovCloudRepositoryManager
// ().isNetworkAvailable()) {
//                                FFUtils.checkNetworkAccess(null);
//                            }}}).retryWhen(new RetryWhenNetworkException(2))//重试逻辑;}}}

主要的重试逻辑

public class RetryWhenNetworkException implements Function<Observable<? extends Throwable>, Observable<?>> {// 可重试次数private int maxConnectCount = 2;// 当前已重试次数private int currentRetryCount = 0;// 重试等待时间private int waitRetryTime = 0;public RetryWhenNetworkException(int maxConnectCount) {this.maxConnectCount = maxConnectCount;}@Overridepublic Observable<?> apply(Observable<? extends Throwable> throwableObservable) throws Exception {// 参数Observable<Throwable>中的泛型 = 上游操作符抛出的异常,可通过该条件来判断异常的类型return throwableObservable.flatMap(new Function<Throwable, ObservableSource<?>>() {@Overridepublic ObservableSource<?> apply(Throwable throwable) throws Exception {// 输出异常信息FLog.d("发生异常 = " + throwable.toString());/*** 需求1:根据异常类型选择是否重试* 即,当发生的异常 = 网络异常 = IO异常 才选择重试*/if (throwable instanceof IOException ) {
//                if (throwable instanceof TimeoutException ) {
//                    FLog.d("属于IO异常,需重试");FLog.d("属于网络异常,需重试");/*** 需求2:限制重试次数* 即,当已重试次数 < 设置的重试次数,才选择重试*/if (currentRetryCount < maxConnectCount) {// 记录重试次数currentRetryCount++;FLog.d("重试次数 = " + currentRetryCount);/*** 需求2:实现重试* 通过返回的Observable发送的事件 = Next事件,从而使得retryWhen()重订阅,最终实现重试功能** 需求3:延迟1段时间再重试* 采用delay操作符 = 延迟一段时间发送,以实现重试间隔设置** 需求4:遇到的异常越多,时间越长* 在delay操作符的等待时间内设置 = 每重试1次,增多延迟重试时间0.5s*/// 设置等待时间waitRetryTime = 500 + currentRetryCount * 500;FLog.d("等待时间 =" + waitRetryTime);return Observable.just(1).delay(waitRetryTime, TimeUnit.MILLISECONDS);} else {// 若重试次数已 > 设置重试次数,则不重试// 通过发送error来停止重试(可在观察者的onError()中获取信息)return Observable.error(new Throwable("重试次数已超过设置次数 = " + currentRetryCount + ",即 不再重试;"+throwable));}}// 若发生的异常不属于I/O异常,则不重试// 通过返回的Observable发送的事件 = Error事件 实现(可在观察者的onError()中获取信息)else {FLog.e(new Throwable("发生了非网络异常(非I/O异常)"));return Observable.error(throwable);}}});}
}

这篇关于Android OKHTTP 网络请求出错重连(结合Retrofit RxJava)--retryWhen命令符的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!


原文地址:
本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.chinasem.cn/article/979235

相关文章

SpringBoot中六种批量更新Mysql的方式效率对比分析

《SpringBoot中六种批量更新Mysql的方式效率对比分析》文章比较了MySQL大数据量批量更新的多种方法,指出REPLACEINTO和ONDUPLICATEKEY效率最高但存在数据风险,MyB... 目录效率比较测试结构数据库初始化测试数据批量修改方案第一种 for第二种 case when第三种

Java docx4j高效处理Word文档的实战指南

《Javadocx4j高效处理Word文档的实战指南》对于需要在Java应用程序中生成、修改或处理Word文档的开发者来说,docx4j是一个强大而专业的选择,下面我们就来看看docx4j的具体使用... 目录引言一、环境准备与基础配置1.1 Maven依赖配置1.2 初始化测试类二、增强版文档操作示例2.

一文详解如何使用Java获取PDF页面信息

《一文详解如何使用Java获取PDF页面信息》了解PDF页面属性是我们在处理文档、内容提取、打印设置或页面重组等任务时不可或缺的一环,下面我们就来看看如何使用Java语言获取这些信息吧... 目录引言一、安装和引入PDF处理库引入依赖二、获取 PDF 页数三、获取页面尺寸(宽高)四、获取页面旋转角度五、判断

Spring Boot中的路径变量示例详解

《SpringBoot中的路径变量示例详解》SpringBoot中PathVariable通过@PathVariable注解实现URL参数与方法参数绑定,支持多参数接收、类型转换、可选参数、默认值及... 目录一. 基本用法与参数映射1.路径定义2.参数绑定&nhttp://www.chinasem.cnbs

JAVA中安装多个JDK的方法

《JAVA中安装多个JDK的方法》文章介绍了在Windows系统上安装多个JDK版本的方法,包括下载、安装路径修改、环境变量配置(JAVA_HOME和Path),并说明如何通过调整JAVA_HOME在... 首先去oracle官网下载好两个版本不同的jdk(需要登录Oracle账号,没有可以免费注册)下载完

Spring StateMachine实现状态机使用示例详解

《SpringStateMachine实现状态机使用示例详解》本文介绍SpringStateMachine实现状态机的步骤,包括依赖导入、枚举定义、状态转移规则配置、上下文管理及服务调用示例,重点解... 目录什么是状态机使用示例什么是状态机状态机是计算机科学中的​​核心建模工具​​,用于描述对象在其生命

Spring Boot 结合 WxJava 实现文章上传微信公众号草稿箱与群发

《SpringBoot结合WxJava实现文章上传微信公众号草稿箱与群发》本文将详细介绍如何使用SpringBoot框架结合WxJava开发工具包,实现文章上传到微信公众号草稿箱以及群发功能,... 目录一、项目环境准备1.1 开发环境1.2 微信公众号准备二、Spring Boot 项目搭建2.1 创建

Java中Integer128陷阱

《Java中Integer128陷阱》本文主要介绍了Java中Integer与int的区别及装箱拆箱机制,重点指出-128至127范围内的Integer值会复用缓存对象,导致==比较结果为true,下... 目录一、Integer和int的联系1.1 Integer和int的区别1.2 Integer和in

SpringSecurity整合redission序列化问题小结(最新整理)

《SpringSecurity整合redission序列化问题小结(最新整理)》文章详解SpringSecurity整合Redisson时的序列化问题,指出需排除官方Jackson依赖,通过自定义反序... 目录1. 前言2. Redission配置2.1 RedissonProperties2.2 Red

IntelliJ IDEA2025创建SpringBoot项目的实现步骤

《IntelliJIDEA2025创建SpringBoot项目的实现步骤》本文主要介绍了IntelliJIDEA2025创建SpringBoot项目的实现步骤,文中通过示例代码介绍的非常详细,对大家... 目录一、创建 Spring Boot 项目1. 新建项目2. 基础配置3. 选择依赖4. 生成项目5.