前言

上一篇文章介绍了RxJava的基础知识和简单实现,篇幅已经比较多了,所以把操作符(Operators)相关的内容放在这一篇。有了上一篇文章的基础,相信会比较容易理解操作符相关的内容了。

操作符(Operators)

拿上一篇文章中的例子HelloWorld为例

1
2
3
4
5
6
7
8
Observable.just("HelloWorld").subscribe(new Action1<String>() {

    public void (String s) {
Log.d(LOG_TAG, s);
}
});

如果希望在输出的字符串后面加上特定字符,比如说”***”,这时候如果我们可以修改Observable对象,那么可以直接在后面加上拼接字符串,程序如下所示

1
2
3
4
5
6
7
8
Observable.just("HelloWorld***").subscribe(new Action1<String>() {

    public void (String s) {
Log.d(LOG_TAG, s);
}
});

但是,如果我们不可以修改Observable对象呢,比如说它是由第三方库提供的呢?或者,这个Observable被多个Observer对象订阅,但是我们只想针对某个Observer做修改呢?这时候可以在Observer中进行修改,程序如下所示

1
2
3
4
5
6
7
8
9
Observable.just("HelloWorld").subscribe(new Action1<String>() {

    public void (String s) {
Log.d(LOG_TAG, s + "***");
}
});

这种方式,看似解决了问题,但是仍然是不够的,此时的Observer不够轻量,根据响应式函数编程的概念,Observer应该做的事情是”响应“,响应Observable发出的事件,而不是修改。如果我们能在Observable发出事件之后,Observer订阅事件之前对“HelloWorld”进行变换,那不就好了!

map操作符

RxJava里面,提供了操作符,用于解决Observable对象变换的问题,用于在Observable和Observer之间修改Observable发出的事件,map操作符就是把一个事件转换成另外一个事件的,程序如下所示

1
2
3
4
5
6
7
8
9
10
11
12
13
Observable.just("HelloWorld").map(new Func1<String, String>() {

    public String (String s) {
return s + "***";
}
}).subscribe(new Action1<String>() { public void (String s) {
Log.d(LOG_TAG, s);
}
});

上面的程序中,转换前后的Observable对象类型是相同的,如果,我们能做到转换前后的数据类型是不同的,是不是会更加酷呢?比如说,Observer不关心返回的字符串,而是想要字符串的hash值,那么就可以这么做

1
2
3
4
5
6
7
8
9
10
11
12
13
Observable.just("HelloWorld").map(new Func1<String, Integer>() {
@Override
public Integer call(String s) {
return s.hashCode();
}
}).subscribe(new Action1<Integer>() {
@Override
public void call(Integer integer) {
Log.d(LOG_TAG, Integer.toString(integer));
}
});

太酷了,我们初始化Observable对象返回的是字符串,最终Observer获取的是Integer。如果简化一下Observer,我们可以再增加一个map操作符,程序如下

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
Observable.just("HelloWorld").map(new Func1<String, Integer>() {
@Override
public Integer call(String s) {
return s.hashCode();
}
}).map(new Func1<Integer, String>() {
@Override
public String call(Integer integer) {
return Integer.toString(integer);
}
}).subscribe(new Action1<String>() {
@Override
public void call(String s) {
Log.d(LOG_TAG, s);
}
});

代码量虽然变多了,但是程序变得更加简洁!

from操作符

这时候,如果我们要发出的事件不再是一个字符串,而是一个字符串列表,如果用现有的知识,程序如下

1
2
3
4
5
6
7
8
9
10
Observable.just(stringList).subscribe(new Action1<List<String>>() {
@Override
public void call(List<String> strings) {
for (String s : strings) {
Log.d(LOG_TAG, s);
}
}
});

这种代码是比较糟糕的,因为在我们的Observer中竟然要做for循环这种复杂的操作,有没有办法简化呢?这时候,from操作符可以帮到我们,程序如下

1
2
3
4
5
6
7
8
Observable.from(stringList).subscribe(new Action1<String>() {
@Override
public void call(String s) {
Log.d(LOG_TAG, s);
}
});

现在的程序看起来很好,for循环消失了,因为from操作符就是用来接收一个集合,然后每次输出一个元素给Observer的!这时,如果我们还要做事件对象的变换,可以这样

1
2
3
4
5
6
7
8
9
10
大专栏  RxJava学习笔记(操作符)">11
12
13
Observable.from(stringList).map(new Func1<String, Integer>() {
@Override
public Integer call(String s) {
return s.hashCode();
}
}).subscribe(new Action1<Integer>() {
@Override
public void call(Integer integer) {
Log.d(LOG_TAG, Integer.toString(integer));
}
});

简洁明了,看着舒服,哈哈!
现在有一个需求,有教师这样的一个数据结构,里面有名字name和课程列表courseList,现在要打印一组老师的名字,程序如下

1
2
3
4
5
6
7
8
Observable.from(teacherList).subscribe(new Action1<Teacher>() {
@Override
public void call(Teacher teacher) {
Log.d(LOG_TAG, teacher.getName());
}
});

flatMap操作符

实现很简单,那么再有一个需求,要打印出每个教师所教授的课程名称呢?(两个需求的区别在于,每个教师只有一个名字,但是可以教授多门课程),我们首先可能会这么写

1
2
3
4
5
6
7
8
9
10
Observable.from(teacherList).subscribe(new Action1<Teacher>() {
@Override
public void call(Teacher teacher) {
for (Course course : teacher.getCourseList()) {
Log.d(LOG_TAG, course.getCourseName());
}
}
});

同样的问题又来了,我们的Observer不够轻量,在里面做了for循环的操作,而且,这时候map和from操作符也不能解决问题,因为map做的是一对一的变换,所以flatMap出场了,程序如下

1
2
3
4
5
6
7
8
9
10
11
12
13
Observable.from(teacherList).flatMap(new Func1<Teacher, Observable<Course>>() {
@Override
public Observable<Course> call(Teacher teacher) {
return Observable.from(teacher.getCourseList());
}
}).subscribe(new Action1<Course>() {
@Override
public void call(Course course) {
Log.d(LOG_TAG, course.getCourseName());
}
});

代码又重新变得简洁明了,实现了一个teacher到多个Course对象的变换!这里需要注意一下,虽然map和flatMap操作符都是把传入参数转换后返回另一个对象,但是flatMap操作符返回的是Observable对象,并且这个 Observable 对象并不是被直接发送到了Observer的回调方法中!flatMap的原理:

  • 使用传入的事件对象创建一个Observable对象
  • 将这个新的Observable对象激活,于是事件由这个新的对象发出
  • 每一个创建出来的Observable对象发送的事件,都被汇入同一个Observable,而这个Observable负责将这些事件统一交给Observer的回调方法

其它操作符

到目前为止,我们已经接触了map、from、flatMap等强大而且常用的操作符,但是RxJava中还有很多其它的操作符:

filter操作符

比如说我们在输出课程名字的时候,不想输出数学这门科目,我们可以这么做

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
Observable.from(teacherList).flatMap(new Func1<Teacher, Observable<Course>>() {
@Override
public Observable<Course> call(Teacher teacher) {
return Observable.from(teacher.getCourseList());
}
}).filter(new Func1<Course, Boolean>() {
@Override
public Boolean call(Course course) {
return !course.equals("数学");
}
}).subscribe(new Action1<Course>() {
@Override
public void call(Course course) {
Log.d(LOG_TAG, course.getCourseName());
}
});

filter操作符会输入和输出相同的对象,并且可以过滤掉不满足条件的。

take操作符

比如说我们在输出课程的时候,课程可能会有很多,我们只想最多输出五门课程,我们可以这么做

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
Observable.from(teacherList).flatMap(new Func1<Teacher, Observable<Course>>() {
@Override
public Observable<Course> call(Teacher teacher) {
return Observable.from(teacher.getCourseList());
}
}).filter(new Func1<Course, Boolean>() {
@Override
public Boolean call(Course course) {
return !course.equals("数学");
}
}).take(5)
.subscribe(new Action1<Course>() {
@Override
public void call(Course course) {
Log.d(LOG_TAG, course.getCourseName());
}
});

当然了,RxJava还有很多好用的操作符,要知道Observable这个类可是有九千多,接近一万行代码呢,更多的内容,就不在这里记录了。

参考文章

最新文章

  1. check_pkg函数解析
  2. 将Vim改造为强大的IDE—Vim集成Ctags/Taglist/Cscope/Winmanager/NERDTree/OmniCppComplete(有图有真相)(转)
  3. 加州大学伯克利分校Stat2.3x Inference 统计推断学习笔记: FINAL
  4. JqueryEasyUI 解决IE下加载时页面错乱的问题 分类: JavaScript JqueryEasyUI 2014-09-20 09:50 545人阅读 评论(1) 收藏
  5. BZOJ 2600: [Ioi2011]ricehub
  6. [纯小白学习OpenCV系列]官方例程00:世界观与方法论
  7. Android 文件访问权限的四种模式
  8. 程序中double类型的数输出为什么要用lf
  9. C语言接口的写法(以toyls命令为例)
  10. 简单解析依赖注入(控制反转)在Spring中的应用
  11. floor舍去法取整
  12. 【转】为什么C++编译器不能支持对模板的分离式编译
  13. [AngularJS] angular-formly: expressionProperties
  14. 在Silverlight中打开网页的几种方法
  15. JS获得URL参数
  16. 【Web探索之旅】第四部分:Web程序员
  17. iOS kvc
  18. [转]Building a Basic Fuzzer with GDB: The Five Minute GDB Scripting Tutorial
  19. 行编辑距离Edit Distance——动态规划
  20. Lambda表达式补充

热门文章

  1. linux中awk的应用
  2. Hard Disk Drive(MBR)
  3. mysql 几个坑
  4. 基于Linux下的C语言项目实战--本地账号管理系统
  5. vue中axios的post请求使用form表单格式发送数据
  6. iTOP-4412开发板qt4文件的移植和修改
  7. day26-socket(server和client通信)
  8. [洛谷P3806] [模板] 点分治1
  9. http 3种web会话管理方式
  10. [LC] 150. Evaluate Reverse Polish Notation