上篇文章分析了subscribeOn的源码逻辑,它的实质就是把上游的Observable.onSubscribe.call(subscriber1)放到了指定的Scheduler线程中执行;本文要分析的observeOn,实质则是在Subscriber的onNext(T value)、onComplete()、onError()中另起了一个线程,在新线程里将数据和通知发射给下游的Subscriber。
还是老办法,将代码层层替换:
observable1.observeOn(AndroidSchedulers.mainThread())
.subscribe(subscriber1);
就等价于:
scheduler = AndroidSchedulers.mainThread();
operator = new OperatorObserveOn(scheduler, delayError, bufferSize);
observable2 = new Observable(new OnSubscribeLift(onSubscribe1, operator));
observable2.subscribe(subscriber1);
简单替换下:
operator = new OperatorObserveOn(scheduler, delayError, bufferSize);
onSubscribe2 = new OnSubscribeLift(onSubscribe1, operator);
observable2.subscribe(subscriber1);
替换subscribe(),得到:
operator = new OperatorObserveOn(scheduler, delayError, bufferSize);
onSubscribe2 = new OnSubscribeLift(onSubscribe1, operator);
subscriber1.onStart();
(observable2.onSubscribe).call(subscriber1);
再次简单替换,得到:
operator = new OperatorObserveOn(scheduler, delayError, bufferSize);
onSubscribe2 = new OnSubscribeLift(onSubscribe1, operator);
subscriber1.onStart();
onSubscribe2.call(subscriber1);
替换最后一句,得到:
operator = new OperatorObserveOn(scheduler, delayError, bufferSize);
onSubscribe2 = new OnSubscribeLift(onSubscribe1, operator);
subscriber1.onStart();
Subscriber super T> st = hook.onLift(operator).call(subscriber1);
st.onStart();
onSubscribe1.call(st);
替换operator.call得到:
operator = new OperatorObserveOn(scheduler, delayError, bufferSize);
onSubscribe2 = new OnSubscribeLift(onSubscribe1, operator);
subscriber1.onStart();
Subscriber super T> st =
{
ObserveOnSubscriber parent = new ObserveOnSubscriber(scheduler, subscriber1, delayError, bufferSize);
parent.init();
return parent;
}
st.onStart();
onSubscribe1.call(st);
至此,可以看到,当上游的数据源发送数据时,会先发给中间侦听者ObserveOnSubscriber相关代码附录如下:
// Observable
public final Observable observeOn(Scheduler scheduler) {
return observeOn(scheduler, RxRingBuffer.SIZE);
}
其中RxRingBuffer.SIZE数值可设置,android平台默认是16.
public final Observable observeOn(Scheduler scheduler, int bufferSize) {
return observeOn(scheduler, false, bufferSize);
}
public final Observable observeOn(Scheduler scheduler, boolean delayError, int bufferSize) {
if (this instanceof ScalarSynchronousObservable) {
return ((ScalarSynchronousObservable)this).scalarScheduleOn(scheduler);
}
return lift(new OperatorObserveOn(scheduler, delayError, bufferSize));
}
public final Observable lift(final Operator extends R, ? super T> operator) {
return new Observable(new OnSubscribeLift(onSubscribe, operator));
}
// OnSubscribeLift
public OnSubscribeLift(OnSubscribe parent, Operator extends R, ? super T> operator) {
this.parent = parent;
this.operator = operator;
}
@Override
public void call(Subscriber super R> o) {
try {
Subscriber super T> st = hook.onLift(operator).call(o);
try {
// new Subscriber created and being subscribed with so 'onStart' it
st.onStart();
parent.call(st);
} catch (Throwable e) {
// localized capture of errors rather than it skipping all operators
// and ending up in the try/catch of the subscribe method which then
// prevents onErrorResumeNext and other similar approaches to error handling
Exceptions.throwIfFatal(e);
st.onError(e);
}
} catch (Throwable e) {
Exceptions.throwIfFatal(e);
// if the lift function failed all we can do is pass the error to the final Subscriber
// as we don't have the operator available to us
o.onError(e);
}
}
// OperatorObserveOn
public Subscriber super T> call(Subscriber super T> child) {
if (scheduler instanceof ImmediateScheduler) {
// avoid overhead, execute directly
return child;
} else if (scheduler instanceof TrampolineScheduler) {
// avoid overhead, execute directly
return child;
} else {
ObserveOnSubscriber parent = new ObserveOnSubscriber(scheduler, child, delayError, bufferSize);
parent.init();
return parent;
}
}
// ObserveOnSubscriber
// static final class ObserveOnSubscriber extends Subscriber implements Action0
public ObserveOnSubscriber(Scheduler scheduler, Subscriber super T> child, boolean delayError, int bufferSize) {
this.child = child;
this.recursiveScheduler = scheduler.createWorker();
}
void init() {
// don't want this code in the constructor because `this` can escape through the
// setProducer call
Subscriber super T> localChild = child;
localChild.setProducer(new Producer() {
@Override
public void request(long n) {
if (n > 0L) {
BackpressureUtils.getAndAddRequest(requested, n);
schedule();
}
}
});
localChild.add(recursiveScheduler);
localChild.add(this);
}
protected void schedule() {
if (counter.getAndIncrement() == 0) {
recursiveScheduler.schedule(this);
}
}
@Override
public void onNext(final T t) {
if (isUnsubscribed() || finished) {
return;
}
if (!queue.offer(on.next(t))) {
onError(new MissingBackpressureException());
return;
}
schedule();
}
@Override
public void onCompleted() {
if (isUnsubscribed() || finished) {
return;
}
finished = true;
schedule();
}
@Override
public void onError(final Throwable e) {
if (isUnsubscribed() || finished) {
RxJavaPlugins.getInstance().getErrorHandler().handleError(e);
return;
}
error = e;
finished = true;
schedule();
}
public void call() {
for (;;) {
long requestAmount = requested.get();
while (requestAmount != currentEmission) {
localChild.onNext(localOn.getValue(v));
}
}
}
// AndroidSchedulers
public static Scheduler mainThread() {
return getInstance().mainThreadScheduler;
}
mainThreadScheduler = new LooperScheduler(Looper.getMainLooper());
// LooperScheduler
public Worker createWorker() {
return new HandlerWorker(handler);
}
// HandlerWorker
public Subscription schedule(Action0 action, long delayTime, TimeUnit unit) {
if (unsubscribed) {
return Subscriptions.unsubscribed();
}
action = hook.onSchedule(action);
ScheduledAction scheduledAction = new ScheduledAction(action, handler);
Message message = Message.obtain(handler, scheduledAction);
message.obj = this; // Used as token for unsubscription operation.
handler.sendMessageDelayed(message, unit.toMillis(delayTime));
if (unsubscribed) {
handler.removeCallbacks(scheduledAction);
return Subscriptions.unsubscribed();
}
return scheduledAction;
}
@Override
public Subscription schedule(final Action0 action) {
return schedule(action, 0, TimeUnit.MILLISECONDS);
}
// ScheduledAction
//final class ScheduledAction implements Runnable, Subscription
@Override public void run() {
try {
action.call();
} catch (Throwable e) {
// nothing to do but print a System error as this is fatal and there is nowhere else to throw this
IllegalStateException ie;
if (e instanceof OnErrorNotImplementedException) {
ie = new IllegalStateException("Exception thrown on Scheduler.Worker thread. Add `onError` handling.", e);
} else {
ie = new IllegalStateException("Fatal Exception thrown on Scheduler.Worker thread.", e);
}
RxJavaPlugins.getInstance().getErrorHandler().handleError(ie);
Thread thread = Thread.currentThread();
thread.getUncaughtExceptionHandler().uncaughtException(thread, ie);
}
}