Subject使用及示例
首先讲一下它是什么有什么用?
Subject可以看成是一个桥梁或者代理,在某些ReactiveX实现中(如RxJava),它同时充当了Observer(观察者)和Observable(被观察者)的角色。因为它是一个Observer,它可以订阅一个或多个Observable;又因为它是一个Observable,它可以转发它收到(Observe)的数据,也可以发射新的数据。
由于一个Subject订阅一个Observable,它可以触发这个Observable开始发射数据(如果那个Observable是”冷”的–就是说,它等待有订阅才开始发射数据)。因此有这样的效果,Subject可以把原来那个”冷”的Observable变成”热”的。
RxJava中常见的Subject有4种,分别是 AsyncSubject、 BehaviorSubject、 PublishSubject、 ReplaySubject。
值得注意的是一定要用Subcect.create()的方式创建并使用,不要用just(T)、from(T)、create(T)创建,否则会导致失效…
AsyncSubject
简单的说使用AsyncSubject无论输入多少参数,永远只输出最后一个参数。也可以这样理解,输出onComplete之前的最近一个参数
AsyncSubject<Integer> asyncSubject = AsyncSubject.create();
asyncSubject.onNext(1);
asyncSubject.onNext(2);
asyncSubject.onNext(3);
asyncSubject.onCompleted();//这里必须调用,否则没有输出
asyncSubject.subscribe(new Action1<Integer>() {
@Override
public void call(Integer s) {
System.out.println("=======:"+s);//输出3
}
});
但是如果因为发生了错误而终止,AsyncSubject将不会发射任何数据,只是简单的向前传递这个错误通知
BehaviorSubject
BehaviorSubject会发送离订阅最近的上一个值以及下面的值,没有上一个值的时候会发送默认值。看图
如果遇到错误会直接中断
// 注意订阅时机,以下这个例子收不到回调
BehaviorSubject<Integer> behaviorSubject = BehaviorSubject.create();
//这里订阅打印1,2,3(前面没有值,发送后面的)
behaviorSubject.onNext(1);
//这里订阅打印1,2,3(前面有个1,后面2,3)
behaviorSubject.onNext(2);
//这里订阅打印2,3(前面是2,后面是3)
behaviorSubject.onNext(3);
//这里订阅打印3(前面是3,后面没值)
behaviorSubject.onCompleted();
//这里订阅没有打印
behaviorSubject.subscribe(new Action1<Integer>() {
@Override
public void call(Integer integer) {
System.out.println("========: "+integer);//不会打印
}
});
PublishSubject
可以说是最正常的Subject,从那里订阅就从那里开始发送数据
PublishSubject<Integer> publishSubject = PublishSubject.create();
//1,2,3
publishSubject.onNext(1);
publishSubject.onNext(2);
//3
publishSubject.onNext(3);
//没有输出
publishSubject.onCompleted();
publishSubject.subscribe(new Action1<Integer>() {
@Override
public void call(Integer integer) {
System.out.println("======: "+integer);//没有输出
}
});
ReplaySubject
无论何时订阅,都会将所有历史订阅内容全部发出。
ReplaySubject<Integer> replaySubject = ReplaySubject.create();
replaySubject.onNext(1);
replaySubject.onNext(2);
replaySubject.onNext(3);
replaySubject.onCompleted();
replaySubject.subscribe(new Action1<Integer>() {
@Override
public void call(Integer integer) {
System.out.println("=========: "+integer);//无论何时订阅,全部都会输出
}
});
下面我们看一个Subject的使用例子,我们要实现这样一个功能,点击按钮增加数字,先看效果图
代码如下:
private TextView mTextView;
private int count = 0;
private PublishSubject<Integer> mSubject;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
mTextView = (TextView) findViewById(R.id.textview);
mSubject = PublishSubject.create();
mSubject.subscribe(new Subscriber<Integer>() {
@Override
public void onCompleted() {
}
@Override
public void onError(Throwable e) {
}
@Override
public void onNext(Integer integer) {
mTextView.setText(String.valueOf(integer));
}
});
}
public void click(View view){
count++;
mSubject.onNext(count);
}
我们每次点击按钮,因为Subject也是观察者,所以我们调用onNext方法,每次把值传过去,这时候,如果进行了订阅,我们在onNext方法中把取出来的值设置给TextView进行显示。