RxJava2.0 操作符(5)—— Error Handle 错误指示符

很多操作符可用于对 Observable 发射的 onError 通知做出响应或者从错误中恢复,例如,你可以:

  1. 捕捉这个错误,切换到一个备用的 Observable 继续发射数据。
  2. 捕捉这个错误然后发射默认值。
  3. 捕捉这个错误并立即尝试重启这个 Observable。
  4. 捕捉这个错误,在一些回退间隔后重启这个 Observable。

Catch — 从 onError 通知中恢复发射数据。
Retry — 如果原始 Observable 遇到错误,重新订阅它期望它能正常终止。

5.1 Catch

从 onError 通知中恢复发射数据。

Catch
Catch

Catch 操作符拦截原始 Observable 的 onError 通知,将它替换为其它的数据项或数据序列,让产生的 Observable 能够正常终止或者根本不终止。
RxJava 将 Catch 实现为三个不同的操作符:onErrorReturn ,onErrorResumeNext,onExceptionResumeNext。

5.1.1 onErrorReturn

让 Observable 遇到错误时发射一个特殊的项并且正常终止。

onErrorReturn
onErrorReturn

示例代码:

Observable.create(new ObservableOnSubscribe<String>() {
        @Override
        public void subscribe(@NonNull ObservableEmitter<String> emitter) throws Exception {
            for (int i = 0; i <= 5; i++) {
                emitter.onNext("" + i);
                if (i == 3) {
                    emitter.onError(new Throwable("nothing"));
                }
            }
            emitter.onComplete();
        }
    }).onErrorReturn(new Function<Throwable, String>() {
        @Override
        public String apply(@NonNull Throwable throwable) throws Exception {
            return "Error Return";
        }
    }).subscribe(new Observer<String>() {
        @Override
        public void onSubscribe(@NonNull Disposable d) {

        }

        @Override
        public void onNext(@NonNull String s) {
            Log.e(TAG, "onNext:" + s);
        }

        @Override
        public void onError(@NonNull Throwable e) {
            Log.e(TAG, "onError" + e.getMessage());
        }

        @Override
        public void onComplete() {
            Log.e(TAG, "onComplete");
        }
    });

输出结果:

onNext:0
onNext:1
onNext:2
onNext:3
onNext:Error Return
onComplete

5.1.2 onErrorResumeNext

让 Observable 在遇到错误时,由另外一个 Observable 代替当前的 Observable 并继续发射数据。

onErrorResumeNext
onErrorResumeNext

它有两种实现

  1. OnErrorResumeNext(Observable);出现错误时取到当前 Observable 并做处理。
  2. OnErrorResumeNext(Function(Throwable,Observable));出现错误时发射另一个 Observable。
    示例代码:
 Observable.create(new ObservableOnSubscribe<String>() {
        @Override
        public void subscribe(@NonNull ObservableEmitter<String> emitter) throws Exception {
            for (int i = 0; i <= 5; i++) {
                emitter.onNext("" + i);
                if (i == 3) {
                    emitter.onError(new Throwable("nothing"));
                }
            }
            emitter.onComplete();
        }
    }).onErrorResumeNext(new Function<Throwable, ObservableSource<? extends String>>() {
        @Override
        public ObservableSource<? extends String> apply(@NonNull Throwable t) throws Exception {
            return Observable.just("hello","world");
        }
    }).subscribe(new Observer<String>() {
        @Override
        public void onSubscribe(@NonNull Disposable d) {

        }

        @Override
        public void onNext(@NonNull String s) {
            Log.e(TAG, "onNext:" + s);
        }

        @Override
        public void onError(@NonNull Throwable e) {
            Log.e(TAG, "onError" + e.getMessage());
        }

        @Override
        public void onComplete() {
            Log.e(TAG, "onComplete");
        }
    });

输出结果:

onNext:0
onNext:1
onNext:2
onNext:3
onNext:hello
onNext:world
onComplete

5.1.3 onExceptionResumeNext

onExceptionResumeNext 类似于 OnErrorResume ,不同之处在于其会对 onError 抛出的数据类型做判断,如果是 Exception,也会使用另外一个 Observable 代替原 Observable 继续发射数据;否则会将错误分发给 Subscriber。

onExceptionResumeNext
onExceptionResumeNext

示例代码:

 Observable.create(new ObservableOnSubscribe<String>() {
        @Override
        public void subscribe(@NonNull ObservableEmitter<String> emitter) throws Exception {
            for (int i = 0; i <= 5; i++) {
                emitter.onNext("" + i);
                if (i == 3) {
                    emitter.onError(new Throwable("error throable"));//发送一个throwable
//                  emitter.onError(new Exception("error exception"));//发送一个exception
                }
            }
            emitter.onComplete();
        }
    }).onExceptionResumeNext(Observable.just("hello","world")).subscribe(new Observer<String>() {
        @Override
        public void onSubscribe(@NonNull Disposable d) {

        }

        @Override
        public void onNext(@NonNull String s) {
            Log.e(TAG, "onNext:" + s);
        }

        @Override
        public void onError(@NonNull Throwable e) {
            Log.e(TAG, "onError:" + e.getMessage());
        }

        @Override
        public void onComplete() {
            Log.e(TAG, "onComplete");
        }
    });

当发送 throwable 时,错误分发给 Subscriber。
输出结果 1:

onNext:0
onNext:1
onNext:2
onNext:3
onError:error throable

当发送 exception 时,使用新的 Observable。
输出结果 2:

onNext:0
onNext:1
onNext:2
onNext:3
onNext:hello
onNext:world
onComplete

5.2 Retry

如果原始 Observable 遇到错误,重新订阅它期望它能正常终止。

Retry
Retry

Retry 操作符不会将原始 Observable 的 onError 通知传递给观察者,它会订阅这个 Observable,再给它一次机会无错误地完成它的数据序列。Retry 总是传递 onNext 通知给观察者,由于重新订阅,可能会造成数据项重复,如上图所示。

RxJava 中的实现为 Retry 和 RetryWhen。

5.2.1 Retry

无论收到多少次 onError 通知,无参数版本的 retry 都会继续订阅并发射原始 Observable。

Retry 的三种实现方法:

  1. retry(long) :会最多重新订阅指定的次数,如果次数超了,它不会尝试再次订阅,它会把最新的一个 onError 通知传递给它的观察者。
  2. retry(): 等价于 retry(Long.MAX_VALUE) ;
  3. retry(BiPredicate) :这个函数返回一个 boolean,如果返回 true,retry 再次订阅原 Observable;如果返回 false,retry 会将最新的一个 onError 通知传递给它的观察者。

示例代码 1:

Observable.create(new ObservableOnSubscribe<String>() {
    @Override
    public void subscribe(@NonNull ObservableEmitter<String> emitter) throws Exception {
        for (int i = 3; i <= 5; i++) {
            emitter.onNext("" + i);
            if (i == 3) {
//                        emitter.onError(new Throwable("error throable"));
                emitter.onError(new Exception("error exception"));
            }
        }
        emitter.onComplete();
    }
}).retry(2)
        .subscribe(new Observer<String>() {
    @Override
    public void onSubscribe(@NonNull Disposable d) {

    }

    @Override
    public void onNext(@NonNull String s) {
        Log.e(TAG, "onNext:" + s);
    }

    @Override
    public void onError(@NonNull Throwable e) {
        Log.e(TAG, "onError:" + e.getMessage());
    }

    @Override
    public void onComplete() {
        Log.e(TAG, "onComplete");
    }
});

输出结果:

onNext:3
onNext:3
onNext:3
onError:error exception

示例代码 2:

//将前一段代码 retry(2) 替换成。两者等价
retry(new BiPredicate<Integer, Throwable>() {
    @Override
    public boolean test(@NonNull Integer integer, @NonNull Throwable throwable) throws Exception {
//          Log.e(TAG, integer + "");
        if (integer >= 2) {
            return false;
        }
        return true;
    }
})

输出结果:

onNext:3
onNext:3
onNext:3
onError:error exception

5.2.1 RetryWhen

RetryWhen
RetryWhen

retryWhen 和 retry 类似,区别是,retryWhen 将 onError 中的 Throwable 传递给一个函数,这个函数产生另一个 Observable,retryWhen 观察它的结果再决定是不是要重新订阅原始的 Observable。如果这个 Observable 发射了一项数据,它就重新订阅,如果这个 Observable 发射的是 onError 通知,它就将这个通知传递给观察者然后终止。

需要注意的是使用 retryWhen 的时候,因为每次重新订阅都会产生错误,所以作为参数的 obserbvable 会不断地发射数据,使用 zipWith 操作符可以限制重新订阅的次数,否则会无限制地重新订阅。

示例代码 2:

Observable.create(new ObservableOnSubscribe<String>() {
        @Override
        public void subscribe(@NonNull ObservableEmitter<String> emitter) throws Exception {
            emitter.onNext("3");
            emitter.onError(new Exception("error exception"));
            emitter.onComplete();
        }
    }).retryWhen(new Function<Observable<Throwable>, ObservableSource<?>>() {
        @Override
        public ObservableSource<?> apply(@NonNull Observable<Throwable> to) throws Exception {
    
            return to.zipWith(Observable.range(1, 4)
            , new BiFunction<Throwable, Integer, String>() {
                @Override
                public String apply(@NonNull Throwable t, @NonNull Integer i) throws Exception {
                    Log.e(TAG, "retryWhen count :" + integer);
                    return "integer - "+integer;
                }
            });
        }
    }).subscribe(new Observer<String>() {
        @Override
        public void onSubscribe(@NonNull Disposable d) {
    
        }
    
        @Override
        public void onNext(@NonNull String s) {
            Log.e(TAG, "onNext:" + s);
        }
    
        @Override
        public void onError(@NonNull Throwable e) {
            Log.e(TAG, "onError:" + e.getMessage());
        }
    
        @Override
        public void onComplete() {
            Log.e(TAG, "onComplete");
        }
    });

输出结果:

onNext:3
retryWhen count :1
onNext:3
retryWhen count :2
onNext:3
retryWhen count :3
onNext:3
retryWhen count :4
onComplete

retry 会将错误分发给观察者,而 retryWhen 会正常结束,并不会讲错误分发出去。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,047评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,807评论 3 386
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,501评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,839评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,951评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,117评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,188评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,929评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,372评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,679评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,837评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,536评论 4 335
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,168评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,886评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,129评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,665评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,739评论 2 351

推荐阅读更多精彩内容