RxJava解析

本文详细解析RxJava的核心思想——链式调用和线程切换。通过分析源码,阐述了基本调用方式、链式调用的实现原理,以及`subscribeOn`和`observeOn`方法在切换线程中的作用。文章通过实例代码解释了如何在不同阶段影响线程执行,并总结了线程切换规则。
摘要由CSDN通过智能技术生成

RxJava解析

RxJava的核心思想是通过链式调用使得异步代码调用变得简洁方便。看了网上其他一些介绍RxJava调用的文章时,对其使用和实现原理还是理解不到位,最后通过分析源码的调用弄明白了链式调用和线程切换的实现方式。本文章的内容分为三个部分:

一、基本的调用方式
二、链式调用
三、线程切换


一、基本的调用方式

链式调用的实现:之所以能实现链式调用,是因为每个函数返回的对象都是同一个类型的对象,一次最简单的RxJava异步调用如下所示::

Observable.create(new Observable.OnSubscribe<String>() {
    @Override
    public void call(Subscriber<? super String> subscriber) {
        subscriber.onNext("hello world!");
        subscriber.onCompleted();
    }
}).subscribe(new Subscriber<String>() {
    @Override
    public void onCompleted() {
        Log.d(TAG, "Completed!");
    }

    @Override
    public void onError(Throwable e) {
        Log.d(TAG, "Error");
    }

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

这个函数有两步构成:
1.Observable.create很好理解,即创建了一个Observable对象,且给该对象的属性OnSubscribe赋值为新建的OnSubscribe对象,在这一时刻什么都没有发生,只是创建了Observable对象。
2.继续调用subscribe方法,看看这个方法的核心代码实现(之后的代码都为省略后的代码):

public Subscription subscribe(Subscriber subscriber, Observable observable) {
    subscriber.onStart();
    onSubscribe.call(subscriber);
    return subscriber;
} 

其实就是回调了该Observable中OnSubscribe的call方法,而在call方法中又回调了订阅者Subscriber的onNext方法,完成对于订阅者的回调。


二、链式调用

看看下面这个列子:

Observable.create(new Observable.OnSubscribe<String>() {
    @Override
    public void call(Subscriber<? super String> subscriber) {
        subscriber.onNext("hello world!");
        subscriber.onCompleted();
    }
}).map(new Func1<String, Integer>() {
    public Integer call(String str) {
        return str.length();
    }
}).subscribe(new Subscriber<Integer>() {
    @Override
    public void onCompleted() {
        Log.d(TAG, "Completed!");
    }
    @Override
    public void onError(Throwable e) {
        Log.d(TAG, "Error");
    }
    @Override
    public void onNext(Integer s) {
        Log.d(TAG, "Item: " + s);
    }
}); 

这个例子与上面例子的唯一区别就是在中间调用了一次map操作,把最初泛型为String的Observable1对象转换成了泛型为Integer的Observable2对象,然后又调用了Observable2对象的subscribe方法。那么这里的核心就是map函数,而map函数的核心是lift函数,该函数在所有中间过程产生Observable的过程中基本上都会被调用,那么看看该函数在该代码例子中的核心实现:

public final  Observable<Integer> lift(final Operator<Integer, String> operator) {
    return new Observable<Integer>(new OnSubscribe<Integer>() {
        @Override
        public void call(Subscriber<Integer> o) {
                Subscriber<String> st = operator.call(o);  // 1
                st.onStart();  // 2
                onSubscribe.call(st);  // 3
        }
    });
} 

可以看出该函数确实创建了一个新的Observable2对象和该对象的OnSubscribe2对象,当调用该对象的subscribe方法时,就会调用该对象的OnSubscribe2对象的call方法,我们来看看该call方法调用设计的三行代码:

1.根据参数operator创建了一个订阅者Subscriber0;
2.调用该Subscriber2的onStart方法;
3.因为lift方法是由Observable1对象调用的,因此这里的onSubscribe是Observable1的onSbuscribe2;说明第一步创建的订阅者是Subscriber0是Observable1的订阅者,是通过Observable1的onSubscribe1调用了这个subscribe0的onNext方法,这里我们就需要看一看,这个订阅者subscribe0做了什么,再看看operator.call的内容:

public Subscriber<String> call(final Subscriber<Integer> o) {
    return new Subscriber<T>(o) {
        @Override
        public void onNext(T t) {
             o.onNext(transformer.call(t));
        }

    };
}

这里的transformer是什么?就是map时传入的Func1对象,它会把onSubscribe1传入的string参数转换为integer,然后再调用参数subscriber2(就是最后一个subscriber调用传入的订阅者)的onNext方法。最终map的调用可以这样理解:

1.为第一个客户端代码create的Observable1对象产生一个subscriber1对象,该subscriber1对象封装了map中的Func1方法;当最终订阅的时候,Observable1的订阅方法完成后,它的onSubscribe1会调用该subscriber1的onNext方法,即Func1的call方法,把string转为integer;

2.同时会产生一个新的Observable2对象,并创建该对象的Onsubscribe2对象,Observable2对象的订阅者为最后subscriber的subscriber2对象,当该Observable2调用subscriber的时候,会调用onSubscribe2的call方法,该方法会先调用产生Observable2对象的Observable1对象的订阅者的回调,即Func1,然后再调用Observable2对象的subscriber2对象的onNext方法。即这条链上的每个Observable2对象订阅时,都会先回调产生它的上一个Observable1对象的订阅者的回调方法,即产生Observable2时为Observalbe1添加的订阅者的回调。即map中的Func1是对create产生的Observable1的观察回调。


二、链式调用

(一)subscribeOn切换线程

让我们先来看看下面的一段代码:

 Observable.create(new Observable.OnSubscribe<String>() {
    @Override
    public void call(Subscriber<? super String> subscriber) {
        subscriber.onNext("hello world!");
        subscriber.onCompleted();
    }
})
.subscribeOn(Schedulers.newThread())
.subscribe(new Subscriber<String>() {
    @Override
    public void onCompleted() {
        Log.d(TAG, "Completed!");
    }
    @Override
    public void onError(Throwable e) {
        Log.d(TAG, "Error");
    }
    @Override
    public void onNext(String s) {
        Log.d(TAG, "Item: " + s);
    }
});

这段代码的作用是将create的Observable1的OnSubscribe的call方法放到下面subscribeOn指定的新线程中执行。看看subscribeOn的调用:

 public final Observable<String> subscribeOn(Scheduler scheduler) {
    return nest().lift(new OperatorSubscribeOn<String>(scheduler));
}

该函数先调用nest函数,此函数也创建了一个新的Observable对象:

 public final Observable<Observable<String>> nest() {
    return ScalarSynchronousObservable.create(this);
}

可以看到该函数产生了一个新的Observable,并把其泛型类型变为了

 public static final <Observable<String>> ScalarSynchronousObservable<Observable<String>> create(Observable<String> t) {
    return new ScalarSynchronousObservable<Observable<String>>(t);
}

private final Observable<String> t;

protected ScalarSynchronousObservable(final Observable<String> t) {
    super(new OnSubscribe<Observable<String>>() {
        @Override
        public void call(Subscriber<Observable<String>> s) {
            s.onNext(t);
            s.onCompleted();
        }

    });
    this.t = t;
}

在ScalarSynchronousObservable的OnSubscribe的call方法中,它没有像之前lift产生的Observable一样,回调产生它的Observable的OnSubscribe的call方法,而是把产生它的Observable交给了它的Subscriber去处理。而该ScalarSynchronousObservable的Subscriber是通过nest之后的lift操作所产生的Subscriber去处理。因此根据subscriberOn的功能可以判断,该lift产生的Subscriber的onNext方法,应该是为第一个创建的Observable创建一个String类型的subscriber,并判断当前的线程是否是需要切换的线程,如果是则执行参数Observable(即创建的第一个Observable对象)的onSubscribe的call回调方法,如果不是,则将该操作加入执行线程池的任务队列,等待该线程池调度执行。

先看看lift产生的Observable对象,跟之前map的lift是同一个函数:

public final  Observable<String> lift(final Operator<String, Observable<String>> operator) {
    return new Observable<String>(new OnSubscribe<String>() {
        @Override
        public void call(Subscriber<String> o) {
                Subscriber<String> st = operator.call(o);  // 1
                st.onStart();  // 2
                onSubscribe.call(st);  // 3
        }
    });
} 

因此,这里产生的subscriber是关键,在ScalarSynchronousObservable的onSubscribe的call方法中,它会把第一个Observable当做参数去处理,我们来看一下这个subscriber的创建,它是由OperatorSubscribeOn的call方法创建的:

return new Subscriber<Observable<T>>(subscriber) {
    @Override
    public void onCompleted() {
    }

    @Override
    public void onError(Throwable e) {
        subscriber.onError(e);
    }

    @Override
    public void onNext(final Observable<T> o) {
        inner.schedule(new Action0() {
            @Override
            public void call() {
                final Thread t = Thread.currentThread();
                o.unsafeSubscribe(new Subscriber<T>(subscriber) {

                    @Override
                    public void onCompleted() {
                        subscriber.onCompleted();
                    }

                    @Override
                    public void onError(Throwable e) {
                        subscriber.onError(e);
                    }

                    @Override
                    public void onNext(T t) {
                        subscriber.onNext(t);
                    }

                    @Override
                    public void setProducer(final Producer producer) {
                        subscriber.setProducer(new Producer() {

                            @Override
                            public void request(final long n) {
                                if (Thread.currentThread() == t) {
                                    producer.request(n);
                                } else {
                                    inner.schedule(new Action0() {
                                        @Override
                                        public void call() {
                                            producer.request(n);
                                        }
                                    });
                                }
                            }

                        });
                    }

                });
            }
        });
    }

}; 

可以看出来,这个订阅方法的onNext中确实是通过创建的inner进行线程调度,然后给最初的Observable对象创建了subscriber对象,以便形成完整的调用链。因此,subscribeOn方法,使得该Observable之前的Observable的订阅和回调方法在该方法制定的线程上执行。

(二)observeOn方法

Observable.create(new Observable.OnSubscribe<String>() {
    @Override
    public void call(Subscriber<? super String> subscriber) {
        subscriber.onNext("hello world!");
        subscriber.onCompleted();
    }
})
.observeOn(Schedulers.io())
.subscribe(new Subscriber<String>() {
    @Override
    public void onCompleted() {
        Log.d(TAG, "Completed!");
    }
    @Override
    public void onError(Throwable e) {
        Log.d(TAG, "Error");
    }
    @Override
    public void onNext(String s) {
        Log.d(TAG, "Item: " + s);
    }
}); 

observeOn方法也是切换线程的操作,调用该方法后,其之后执行的回调方法会在该方法制定的线程上操作。因此,我们猜想observeOn是直接进行了lift操作进行线程变换,然后继续往下调用,是否如此呢,我们来看看observeOn的实现:

public final Observable<String> observeOn(Scheduler scheduler) {
    return lift(new OperatorObserveOn<String>(scheduler));
}

看看OperatorObserveOn的call方法:

public Subscriber<String> call(Subscriber<String> child) {
    if (scheduler instanceof ImmediateScheduler) {
        return child;
    } else if (scheduler instanceof TrampolineScheduler) {
        return child;
    } else {
        ObserveOnSubscriber<String> parent = new ObserveOnSubscriber<String>(scheduler, child);
        parent.init();
        return parent;
    }
} 

前两个if-else判断,应该是判断当前的线程如果与指定的线程是同一个,则直接调用最终subscriber的onNext方法,不进行线程切换,如果需要切换线程,则创建一个ObserveOnSubscriber对象,在该对象的onNext方法中,会在指定的线程中调度执行其child(即最终传入的subscriber)的回调方法。

总结,对于subscribeOn方法,第一个出现的subscribeOn方法制定的线程,会制定它调用之前的(若它之前还有observeOn方法,会影响第一个出现的observeOn方法之前的)订阅和所有回调方法的执行线程,而observeOn方法会影响它之后出现的回调方法的执行线程。

最后再举一个例子说明,对于下面的代码调用:

Observable.create(new Observable.OnSubscribe<String>() {
    @Override
    public void call(Subscriber<? super String> subscriber) {
        Log.i(TAG, "0 CURRENT THREAD = " + Thread.currentThread().getName());
        subscriber.onNext("hello world!");
        subscriber.onCompleted();
    }
})
.observeOn(Schedulers.io())
.map(new Func1<String, String>() {
    @Override
    public String call(String s) {
        Log.i(TAG, "1 CURRENT THREAD = " + Thread.currentThread().getName());
        return s;
    }
})
.subscribeOn(Schedulers.computation())
.observeOn(Schedulers.newThread())
.subscribe(new Subscriber<String>() {
    @Override
    public void onCompleted() {
        Log.d(TAG, "Completed!");
    }
    @Override
    public void onError(Throwable e) {
        Log.d(TAG, "Error");
    }
    @Override
    public void onNext(String s) {
        Log.i(TAG, "2 CURRENT THREAD = " + Thread.currentThread().getName());
        Log.d(TAG, "Item: " + s);
    }
}); 

0处线程应该是在computation线程,1处线程应该是在io线程,2处线程应该是在new线程,执行结果如下:

这里写图片描述

  • 关于 RxJava 用法,参考 这儿.

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值