RxJava 和 RxAndroid 一 基础使用
1、RxJava 项目地址
https://github.com/ReactiveX/RxJava
2、RxAndroid 项目地址
https://github.com/ReactiveX/RxAndroid
3、RxJava 和 RxAndroid 的关系
RxAndroid是RxJava的一个针对Android平台的扩展,主要用于 Android 开发
4、RxJava和EventBus的区别?
RxAndroid完全可以替代AsycTask来完成各种异步操作,而且还有BindActivity和BindFragment方法,你根本不需要考虑异步操作时的Activity和Fragment的生命周期问题,还有更加强大的的链式调用,可以使程序很简洁。 推荐使用RxAndroid。
5、RxAndroid的使用方法
compile 'io.reactivex:rxandroid:1.2.0'
6、如何查看RxAndroid最新版本?
http://search.maven.org/#search%7Cga%7C1%7Ca%3A%22rxandroid%22
7、RxAndroid具体使用方法
http://gank.io/post/560e15be2dca930e00da1083#toc_14
http://blog.csdn.net/theone10211024/article/details/50435325
http://huxian99.github.io/tags/RxJava/
https://github.com/mcxiaoke/RxDocs
8、创建观察者
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
|
package
lib.com.myapplication;
import
android.support.v7.app.AppCompatActivity;
import
android.os.Bundle;
import
rx.Observer;
import
rx.Subscriber;
public
class
Main2Activity
extends
AppCompatActivity {
@Override
protected
void
onCreate(Bundle savedInstanceState) {
super
.onCreate(savedInstanceState);
setContentView(R.layout.activity_main2);
//创建观察者 2 种方法
Observer<String> observer =
new
Observer<String>() {
@Override
public
void
onCompleted() {
}
@Override
public
void
onError(Throwable e) {
}
@Override
public
void
onNext(String s) {
}
} ;
// Subscriber 继承 Observer ,对Observer类做了扩展
Subscriber<String> subscriber =
new
Subscriber<String>() {
@Override
public
void
onCompleted() {
}
@Override
public
void
onError(Throwable e) {
}
@Override
public
void
onNext(String s) {
}
} ;
}
}
|
- 从上文可以看到,Subscriber继承Observer, 只是 Subscriber对Observer做了一些扩展。Subscriber的使用和Observer完全一样。
- Subscriber 多了一个 onStart 方法
onStart()
: 这是Subscriber
增加的方法。它会在 subscribe 刚开始,而事件还未发送之前被调用,可以用于做一些准备工作,例如数据的清零或重置。这是一个可选方法,默认情况下它的实现为空。需要注意的是,如果对准备工作的线程有要求(例如弹出一个显示进度的对话框,这必须在主线程执行),onStart()
就不适用了,因为它总是在 subscribe 所发生的线程被调用,而不能指定线程。要在指定的线程来做准备工作,可以使用doOnSubscribe()
方法,具体可以在后面的文中看到。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
|
//
Subscriber<String> subscriber =
new
Subscriber<String>() {
@Override
public
void
onCompleted() {
}
@Override
public
void
onError(Throwable e) {
}
@Override
public
void
onNext(String s) {
}
@Override
public
void
onStart() {
super
.onStart();
}
} ;
|
9、创建被观察者
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
|
//create方式
Observable<String> observable = Observable.create(
new
Observable.OnSubscribe<String>() {
@Override
public
void
call(Subscriber<?
super
String> subscriber) {
subscriber.onNext(
"aa"
) ;
subscriber.onNext(
"bb"
) ;
subscriber.onNext(
"cc"
) ;
subscriber.onCompleted();
}
});
//just方式 最多支持10个数据
Observable<String> observable1 = Observable.just(
"aa"
,
"bb"
,
"cc"
) ;
// 将会依次调用:
// onNext("aa");
// onNext("bb");
// onNext("cc");
// onCompleted();
//from方式
//1:集合
List<String> list =
new
ArrayList<>() ;
list.add(
"aa"
) ;
list.add(
"bb"
) ;
list.add(
"cc"
) ;
Observable<String> observable2 = Observable.from( list ) ;
//2:数组
String[] words = {
"aa"
,
"bb"
,
"cc"
};
Observable<String> observable3 = Observable.from( words ) ;
|
- Call()方法:当
Observable
被订阅的时候,OnSubscribe
的call()
方法会自动被调用,事件序列就会依照设定依次触发(对于上面的代码,就是观察者Subscriber
将会被调用三次onNext()
和一次onCompleted()
)。这样,由被观察者调用了观察者的回调方法,就实现了由被观察者向观察者的事件传递,即观察者模式。
10、订阅
由于观察者可以由两种方式被创建,所以订阅的方式也有两种
1
2
|
observable.subscribe(
observable.subscribe(
|
-
Observable.subscribe(Subscriber)
的内部实现是这样的(仅核心代码):1234567//
注意:这不是 subscribe() 的源码,而是将源码中与性能、兼容性、扩展性有关的代码剔除后的核心代码。//
如果需要看源码,可以去 RxJava 的 GitHub 仓库下载。public
Subscription subscribe(Subscriber subscriber) {
subscriber.onStart();
onSubscribe.call(subscriber);
return
subscriber;
}
- 在subscribe() 中,首先会调用 onStart() 方法,这个方法前文已经介绍了,是可选的。接着会调用 call()方法,我们已经分析了在call()方法中会调用多次 onNext() ,最后调用 onCompleted().看到这里你就会突然明白原来subscribe() 方法其实相当于依次执行了:onStart() --> onNext()--> onCompleted()
- 从这也可以看出,在 RxJava 中,
Observable
并不是在创建的时候就立即开始发送事件,而是在它被订阅的时候,即当subscribe()
方法执行的时候。 Observer
和Subscriber
具有相同的角色,而且Observer
在subscribe()
过程中最终会被转换成Subscriber
对象- 将传入的
Subscriber
作为Subscription
返回。这是为了方便unsubscribe()
.
11、RxBus
你是否听说过EventBus , 他是android 中的事件总线。用rxjava同样可以实现android的事件总线功能,也就是RxBus.
关于rxbus 的基本说明在这里 http://nerds.weddingpartyapp.com/tech/2014/12/24/implementing-an-event-bus-with-rxjava-rxbus/
然而这并没有什么卵用 !
下面是RxBus的封装版
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
|
package
lib.com.myapplication;
import
android.support.annotation.NonNull;
import
android.util.Log;
import
java.util.ArrayList;
import
java.util.Collection;
import
java.util.List;
import
java.util.concurrent.ConcurrentHashMap;
import
rx.Observable;
import
rx.subjects.PublishSubject;
import
rx.subjects.Subject;
/**
* Created by ${zyj} on 2016/5/6.
*/
public
class
RxBus {
private
static
final
String TAG = RxBus.
class
.getSimpleName();
private
static
RxBus instance;
public
static
boolean
DEBUG =
false
;
public
static
RxBus get() {
if
(instance ==
null
) {
synchronized
(RxBus.
class
) {
if
(instance ==
null
) {
instance =
new
RxBus();
}
}
}
return
instance;
}
private
RxBus() {
}
private
ConcurrentHashMap<Object, List<Subject>> subjectMapper =
new
ConcurrentHashMap<>();
@SuppressWarnings
(
"unchecked"
)
public
<T> Observable<T> register(
@NonNull
Object tag,
@NonNull
Class<T> clazz) {
List<Subject> subjectList = subjectMapper.get(tag);
if
(
null
== subjectList) {
subjectList =
new
ArrayList<>();
subjectMapper.put(tag, subjectList);
}
Subject<T, T> subject;
subjectList.add(subject = PublishSubject.create());
if
(DEBUG) Log.d(TAG,
"[register]subjectMapper: "
+ subjectMapper);
return
subject;
}
public
void
unregister(
@NonNull
Object tag,
@NonNull
Observable observable) {
List<Subject> subjects = subjectMapper.get(tag);
if
(
null
!= subjects) {
if
( observable !=
null
&& subjects.contains( observable )){
subjects.remove((Subject) observable);
}
if
(isEmpty(subjects)) {
subjectMapper.remove(tag);
}
}
if
(DEBUG) Log.d(TAG,
"[unregister]subjectMapper: "
+ subjectMapper);
}
public
void
post(
@NonNull
Object content) {
post( content.getClass().getName(), content);
}
@SuppressWarnings
(
"unchecked"
)
public
void
post(
@NonNull
Object tag,
@NonNull
Object content) {
List<Subject> subjectList = subjectMapper.get(tag);
if
(!isEmpty(subjectList)) {
for
(Subject subject : subjectList) {
subject.onNext(content);
}
}
if
(DEBUG) Log.d(TAG,
"[send]subjectMapper: "
+ subjectMapper);
}
private
boolean
isEmpty(Collection collection) {
return
null
== collection || collection.isEmpty();
}
}
|
RxBus的使用
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
|
package
lib.com.myapplication;
import
android.os.Bundle;
import
android.support.v7.app.AppCompatActivity;
import
rx.Observable;
import
rx.functions.Action1;
public
class
Activity1
extends
AppCompatActivity {
String tag =
"tag"
;
Observable<String> ob ;
@Override
protected
void
onCreate(Bundle savedInstanceState) {
super
.onCreate(savedInstanceState);
setContentView(R.layout.activity1);
//创建被观察者
ob = RxBus.get().register( tag , String.
class
) ;
//订阅观察事件
ob.subscribe(
new
Action1<String>() {
@Override
public
void
call(String s) {
System.out.println(
"fff-- "
+ s );
}
}) ;
//发送内容
RxBus.get().post( tag ,
"我是内容"
);
}
@Override
protected
void
onDestroy() {
super
.onDestroy();
//取消订阅
RxBus.get().unregister( tag , ob );
}
}
|
- 在Activity销毁的时候,要取消订阅服务 。 否则 post() 次数会随着post()调用逐渐增加
- 除了上面的简单使用外,还可以使用 Schedulers 、AndroidSchedulers 进行线程切换
RxJava 和 RxAndroid 二 ——操作符的使用
1、merge操作符,合并观察对象
19 List<String> list1 = new ArrayList<>() ;
20 List<String> list2 = new ArrayList<>() ;
21
22 list1.add( "1" ) ;
23 list1.add( "2" ) ;
24 list1.add( "3" ) ;
25
26 list2.add( "a" ) ;
27 list2.add( "b" ) ;
28 list2.add( "c" ) ;
29
30 Observable observable1 = Observable.from( list1 ) ;
31 Observable observable2 = Observable.from( list2 ) ;
32
33 //合并数据 先发送observable2的全部数据,然后发送 observable1的全部数据
34 Observable observable = Observable.merge( observable2 , observable1 ) ;
35
36 observable.subscribe(new Action1() {
37 @Override
38 public void call(Object o) {
39 System.out.println( "rx-- " + o );
40 }
41 }) ;
42
运行结果
2、zip 操作符,合并多个观察对象的数据。并且允许 Func2()函数重新发送合并后的数据
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
|
List<String>
new
ArrayList<>() ;
List<String> list2 =
new
ArrayList<>() ;
list1.add(
"1"
) ;
list1.add(
"2"
) ;
list1.add(
"3"
) ;
list2.add(
"a"
) ;
list2.add(
"b"
) ;
list2.add(
"c"
) ;
list2.add(
"d"
) ;
Observable observable1 = Observable.from( list1 ) ;
Observable observable2 = Observable.from( list2 ) ;
Observable observable3 = Observable.zip(observable1, observable2,
new
Func2<String , String , String >() {
@Override
public
String call(String s1 , String s2 ) {
return
s1 + s2 ;
}
}) ;
observable3.subscribe(
new
Action1() {
@Override
public
void
call(Object o) {
System.out.println(
"zip-- "
+ o );
}
}) ;
|
运行效果:从效果图上可以看出,合并两个的观察对象数据项应该是相等的;如果出现了数据项不等的情况,合并的数据项以最小数据队列为准。
3、scan累加器操作符的使用
1
2
3
4
5
6
7
8
9
10
11
12
13
|
Observable
1
,
2
,
3
,
4
,
5
) ;
observable.scan(
new
Func2<Integer,Integer,Integer>() {
@Override
public
Integer call(Integer o, Integer o2) {
return
o + o2 ;
}
})
.subscribe(
new
Action1() {
@Override
public
void
call(Object o) {
System.out.println(
"scan-- "
+ o );
}
}) ;
|
运行效果:
第一次发射得到1,作为结果与2相加;发射得到3,作为结果与3相加,以此类推,打印结果:
4、filter 过滤操作符的使用
1
2
3
4
5
6
7
8
9
10
11
12
13
14
|
Observable
1
,
2
,
3
,
4
,
5
,
6
,
7
) ;
observable.filter(
new
Func1<Integer , Boolean>() {
@Override
public
Boolean call(Integer o) {
//数据大于4的时候才会被发送
return
o >
4
;
}
})
.subscribe(
new
Action1() {
@Override
public
void
call(Object o) {
System.out.println(
"filter-- "
+ o );
}
}) ;
|
运行效果
5、 消息数量过滤操作符的使用
- take :取前n个数据
- takeLast:取后n个数据
- first 只发送第一个数据
- last 只发送最后一个数据
- skip() 跳过前n个数据发送后面的数据
- skipLast() 跳过最后n个数据,发送前面的数据
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
|
//take
Observable observable = Observable.just(
1
,
2
,
3
,
4
,
5
,
6
,
7
) ;
observable.take(
3
)
.subscribe(
new
Action1() {
@Override
public
void
call(Object o) {
System.out.println(
"take-- "
+ o );
}
}) ;
//takeLast 发送最后三个数据
Observable observable2 = Observable.just(
1
,
2
,
3
,
4
,
5
,
6
,
7
) ;
observable2.takeLast(
3
)
.subscribe(
new
Action1() {
@Override
public
void
call(Object o) {
System.out.println(
"takeLast-- "
+ o );
}
}) ;
//first 只发送第一个数据
Observable observable3 = Observable.just(
1
,
2
,
3
,
4
,
5
,
6
,
7
) ;
observable3.first()
.subscribe(
new
Action1() {
@Override
public
void
call(Object o) {
System.out.println(
"first-- "
+ o );
}
}) ;
//last 只发送最后一个数据
Observable observable4 = Observable.just(
1
,
2
,
3
,
4
,
5
,
6
,
7
) ;
observable4.last()
.subscribe(
new
Action1() {
@Override
public
void
call(Object o) {
System.out.println(
"last-- "
+ o );
}
}) ;
//skip() 跳过前2个数据发送后面的数据
Observable observable5 = Observable.just(
1
,
2
,
3
,
4
,
5
,
6
,
7
) ;
observable5.skip(
2
)
.subscribe(
new
Action1() {
@Override
public
void
call(Object o) {
System.out.println(
"skip-- "
+ o );
}
}) ;
//skipLast() 跳过最后两个数据,发送前面的数据
Observable observable6 = Observable.just(
1
,
2
,
3
,
4
,
5
,
6
,
7
) ;
observable5.skipLast(
2
)
.subscribe(
new
Action1() {
@Override
public
void
call(Object o) {
System.out.println(
"skipLast-- "
+ o );
}
}) ;
|
效果图
6、elementAt 、elementAtOrDefault
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
|
//elementAt()
//如果该序号大于数据序列中的最大序列号,则会抛出异常,程序崩溃
//所以在用elementAt操作符的时候,要注意判断发送的数据序列号是否越界
Observable observable7 = Observable.just(
1
,
2
,
3
,
4
,
5
,
6
,
7
) ;
observable7.elementAt(
3
)
.subscribe(
new
Action1() {
@Override
public
void
call(Object o) {
System.out.println(
"elementAt-- "
+ o );
}
}) ;
//elementAtOrDefault( int n , Object default ) 发送数据序列中第n个数据 ,序列号从0开始。
//如果序列中没有该序列号,则发送默认值
Observable observable9 = Observable.just(
1
,
2
,
3
,
4
,
5
) ;
observable9.elementAtOrDefault(
8
,
666
)
.subscribe(
new
Action1() {
@Override
public
void
call(Object o) {
System.out.println(
"elementAtOrDefault-- "
+ o );
}
}) ;
|
运行结果
7、startWith() 插入数据
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
|
//插入普通数据
//startWith
Observable
"aa"
,
"bb"
,
"cc"
) ;
observable
.startWith(
"11"
,
"22"
)
.subscribe(
new
Action1() {
@Override
public
void
call(Object o) {
System.out.println(
"startWith-- "
+ o );
}
}) ;
//插入Observable对象
List<String>
new
ArrayList<>() ;
list.add(
"ww"
) ;
list.add(
"tt"
) ;
observable.startWith(
.subscribe(
new
Action1() {
@Override
public
void
call(Object o) {
System.out.println(
"startWith2 -- "
+ o );
}
}) ;
|
运行结果
8、delay操作符,延迟数据发送
1
2
3
4
5
6
7
8
9
10
|
Observable<String>
"1"
,
"2"
,
"3"
,
"4"
,
"5"
,
"6"
,
"7"
,
"8"
) ;
//延迟数据发射的时间,仅仅延时一次,也就是发射第一个数据前延时。发射后面的数据不延时
observable.delay(
3
, TimeUnit.SECONDS )
//延迟3秒钟
.subscribe(
new
Action1() {
@Override
public
void
call(Object o) {
System.out.println(
"delay-- "
+ o);
}
}) ;
|
9、Timer 延时操作符的使用
使用场景:xx秒后,执行xx
1
2
3
4
5
6
7
8
9
10
|
//5秒后输出
Observable.timer(
5
, TimeUnit.SECONDS )
.observeOn(AndroidSchedulers.mainThread() )
.subscribe(
new
Action1<Long>() {
@Override
public
void
call(Long aLong) {
System.out.println(
"timer--hello world "
+ aLong );
findViewById( R.id.image).setVisibility(View.VISIBLE );
}
}) ;
|
timer 返回一个 Observable , 它在延迟一段给定的时间后发射一个简单的数字0
timer 操作符默认在computation调度器上执行,当然也可以用 Scheduler 在定义执行的线程。
delay 、timer 总结:
- 相同点:delay 、 timer 都是延时操作符。
- 不同点:delay 延时一次,延时完成后,可以连续发射多个数据。timer延时一次,延时完成后,只发射一次数据。
10、interval 轮询操作符,循环发送数据,数据从0开始递增
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
|
package
app.com.myapplication;
import
android.support.v7.app.AppCompatActivity;
import
android.os.Bundle;
import
java.util.concurrent.TimeUnit;
import
rx.Observable;
import
rx.Subscription;
import
rx.functions.Action1;
public
class
IntervalActivity
extends
AppCompatActivity {
Subscription subscription ;
@Override
protected
void
onCreate(Bundle savedInstanceState) {
super
.onCreate(savedInstanceState);
setContentView(R.layout.activity_interval);
//参数一:延迟时间 参数二:间隔时间 参数三:时间颗粒度
Observable observable = Observable.interval(
3000
,
3000
, TimeUnit.MILLISECONDS) ;
subscription = observable.subscribe(
new
Action1() {
@Override
public
void
call(Object o) {
System.out.println(
"interval- "
+ o );
}
}) ;
}
@Override
protected
void
onDestroy() {
super
.onDestroy();
if
( subscription !=
null
){
subscription.unsubscribe();
}
}
}
|
11、doOnNext() 操作符,在每次 OnNext() 方法被调用前执行
使用场景:从网络请求数据,在数据被展示前,缓存到本地
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
|
Observable
"1"
,
"2"
,
"3"
,
"4"
) ;<br>
observable.doOnNext(
new
Action1() {
@Override
public
void
call(Object o) {
System.out.println(
"doOnNext--缓存数据"
+ o );
}
})
.subscribe(
new
Observer() {
@Override
public
void
onCompleted() {
}
@Override
public
void
onError(Throwable e) {
}
@Override
public
void
onNext(Object o) {
System.out.println(
"onNext--"
+ o );
}
}) ;
|
12、Buffer 操作符
- Buffer( int n ) 把n个数据打成一个list包,然后再次发送。
- Buffer( int n , int skip) 把n个数据打成一个list包,然后跳过第skip个数据。
使用场景:一个按钮每点击3次,弹出一个toast
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
|
List<String>
new
ArrayList<>();
for
(
int
i =
1
; i <
10
; i++) {
list.add(
""
+ i);
}
Observable<String> observable = Observable.from(list);
observable
.buffer(
2
)
//把每两个数据为一组打成一个包,然后发送
.subscribe(
new
Action1<List<String>>() {
@Override
public
void
call(List<String> strings) {
System.out.println(
"buffer---------------"
);
Observable.from( strings ).subscribe(
new
Action1<String>() {
@Override
public
void
call(String s) {
System.out.println(
"buffer data --"
+ s );
}
}) ;
}
});
|
例子2:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
|
//第1、2
observable.buffer(
2
,
3
)
//把每两个数据为一组打成一个包,然后发送。第三个数据跳过去
.subscribe(
new
Action1<List<String>>() {
@Override
public
void
call(List<String> strings) {
System.out.println(
"buffer22---------------"
);
Observable.from( strings ).subscribe(
new
Action1<String>() {
@Override
public
void
call(String s) {
System.out.println(
"buffer22 data --"
+ s );
}
}) ;
}
}) ;
|
13、throttleFirst 操作符
在一段时间内,只取第一个事件,然后其他事件都丢弃。
使用场景:1、button按钮防抖操作,防连续点击 2、百度关键词联想,在一段时间内只联想一次,防止频繁请求服务器
Observable.interval( 1 , TimeUnit.SECONDS)
.throttleFirst( 3 , TimeUnit.SECONDS )
.subscribe(new Action1<Long>() {
@Override
public void call(Long aLong) {
System.out.println( "throttleFirst--" + aLong );
}
}) ;
这段代码,是循环发送数据,每秒发送一个。throttleFirst( 3 , TimeUnit.SECONDS ) 在3秒内只取第一个事件,其他的事件丢弃。
运行结果
14、distinct 过滤重复的数据
List<String> list = new ArrayList<>() ;
list.add( "1" ) ;
list.add( "2" ) ;
list.add( "1" ) ;
list.add( "3" ) ;
list.add( "4" ) ;
list.add( "2" ) ;
list.add( "1" ) ;
list.add( "1" ) ;
Observable.from( list )
.distinct()
.subscribe(new Action1<String>() {
@Override
public void call(String s) {
System.out.println( "distinct--" + s );
}
}) ;
从结果可以看出,重复的数据已经被过滤掉了
distinctUntilChanged() 过滤连续重复的数据
List<String> list = new ArrayList<>() ;
list.add( "1" ) ;
list.add( "2" ) ;
list.add( "1" ) ;
list.add( "3" ) ;
list.add( "4" ) ;
list.add( "4" ) ;
list.add( "2" ) ;
list.add( "1" ) ;
list.add( "1" ) ;
Observable.from( list )
.distinctUntilChanged()
.subscribe(new Action1<String>() {
@Override
public void call(String s) {
System.out.println( "distinctUntilChanged--" + s );
}
}) ;
运行结果
从结果可以看出,连续重复的数据已经被过滤掉了
15、debounce() 操作符
一段时间内没有变化,就会发送一个数据。
使用场景:百度关键词联想提示。在输入的过程中是不会从服务器拉数据的。当输入结束后,在400毫秒没有输入就会去获取数据。
避免了,多次请求给服务器带来的压力.
16、doOnSubscribe()
使用场景: 可以在事件发出之前做一些初始化的工作,比如弹出进度条等等
注意:
1、doOnSubscribe() 默认运行在事件产生的线程里面,然而事件产生的线程一般都会运行在 io 线程里。那么这个时候做一些,更新UI的操作,是线程不安全的。
所以如果事件产生的线程是io线程,但是我们又要在doOnSubscribe() 更新UI , 这时候就需要线程切换。
2、如果在 doOnSubscribe()
之后有 subscribeOn()
的话,它将执行在离它最近的 subscribeOn()
所指定的线程。
3、 subscribeOn() 事件产生的线程 ; observeOn() : 事件消费的线程
Observable.create(onSubscribe)
.subscribeOn(Schedulers.io())
.doOnSubscribe(new Action0() {
@Override
public void call() {
progressBar.setVisibility(View.VISIBLE); // 需要在主线程执行
}
})
.subscribeOn(AndroidSchedulers.mainThread()) // 指定主线程
.observeOn(AndroidSchedulers.mainThread())
.subscribe(subscriber);
17、range 操作符的使用
首先看range 方法的源码
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
|
public
static
Observable<Integer> range(
int
start,
int
count) {
if
(count <
0
) {
throw
new
IllegalArgumentException(
"Count can not be negative"
);
}
if
(count ==
0
) {
return
Observable.empty();
}
if
(start > Integer.MAX_VALUE - count +
1
) {
throw
new
IllegalArgumentException(
"start + count can not exceed Integer.MAX_VALUE"
);
}
if
(count ==
1
) {
return
Observable.just(start);
}
return
Observable.create(
new
OnSubscribeRange(start, start + (count -
1
)));
}
//可以通过第三个参数控制range执行的线程
public
static
Observable<Integer> range(
int
start,
int
count, Scheduler scheduler) {
return
range(start, count).subscribeOn(scheduler);
}
|
Range操作符发射一个范围内的有序整数序列,你可以指定范围的起始和长度。
RxJava将这个操作符实现为range
函数,它接受两个参数,一个是范围的起始值,一个是范围的数据的数目。如果你将第二个参数设为0,将导致Observable不发射任何数据(如果设置为负数,会抛异常)。
range
默认不在任何特定的调度器上执行。有一个变体可以通过可选参数指定Scheduler。
例子
1
2
3
4
5
6
7
|
Observable.range(
10
,
3
)
.subscribe(
new
Action1<Integer>() {
@Override
public
void
call(Integer integer) {
Log.v(
"rx_range "
,
""
+ integer ) ;
}
}) ;
|
结果
/rx_range: 10
/rx_range: 11
/rx_range: 12
18、defer 操作符
例子
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
|
package
app.com.myapplication;
import
android.support.v7.app.AppCompatActivity;
import
android.os.Bundle;
import
android.util.Log;
import
rx.Observable;
import
rx.functions.Action1;
import
rx.functions.Func0;
public
class
DeferActivity
extends
AppCompatActivity {
String i =
"10"
;
@Override
protected
void
onCreate(Bundle savedInstanceState) {
super
.onCreate(savedInstanceState);
setContentView(R.layout.activity_defer);
i =
"11 "
;
Observable<String> defer = Observable.defer(
new
Func0<Observable<String>>() {
@Override
public
Observable<String> call() {
return
Observable.just( i ) ;
}
}) ;
Observable test = Observable.just( i ) ;
i =
"12"
;
defer.subscribe(
new
Action1<String>() {
@Override
public
void
call(String s) {
Log.v(
"rx_defer "
,
""
+ s ) ;
}
}) ;
test.subscribe(
new
Action1() {
@Override
public
void
call(Object o) {
Log.v(
"rx_just "
,
""
+ o ) ;
}
}) ;
}
}
|
结果
/rx_defer: 12
/rx_just: 11
- 可以看到,just操作符是在创建Observable就进行了赋值操作,而defer是在订阅者订阅时才创建Observable,此时才进行真正的赋值操作。
Defer
操作符会一直等待直到有观察者订阅它,然后它使用Observable工厂方法生成一个Observable。它对每个观察者都这样做,因此尽管每个订阅者都以为自己订阅的是同一个Observable,事实上每个订阅者获取的是它们自己的单独的数据序列。- 在某些情况下,等待直到最后一分钟(就是知道订阅发生时)才生成Observable可以确保Observable包含最新的数据。