一、Scheduler线程切换

这种场景经常会在“后台线程取数据,主线程展示”的模式中看见

  1. Observable.just(1, 2, 3, 4)
  2. .subscribeOn(Schedulers.io()) // 指定 subscribe() 发生在 IO 线程
  3. .observeOn(AndroidSchedulers.mainThread()) // 指定 Subscriber 的回调发生在主线程
  4. .subscribe(new Action1<Integer>() {
  5. @Override
  6. public void call(Integer number) {
  7. Log.d(tag, "number:" + number);
  8. }
  9. });

二、使用debounce做textSearch

用简单的话讲就是当N个结点发生的时间太靠近(即发生的时间差小于设定的值T),debounce就会自动过滤掉前N-1个结点。

比如在做百度地址联想的时候,可以使用debounce减少频繁的网络请求。避免每输入(删除)一个字就做一次联想

  1. RxTextView.textChangeEvents(inputEditText)
  2. .debounce(400, TimeUnit.MILLISECONDS)
  3. .observeOn(AndroidSchedulers.mainThread())
  4. .subscribe(new Observer<TextViewTextChangeEvent>() {
  5. @Override
  6. public void onCompleted() {
  7. log.d("onComplete");
  8. }
  9. @Override
  10. public void onError(Throwable e) {
  11. log.d("Error");
  12. }
  13. @Override
  14. public void onNext(TextViewTextChangeEvent onTextChangeEvent) {
  15. log.d(format("Searching for %s", onTextChangeEvent.text().toString()));
  16. }
  17. });
三、Retrofit结合RxJava做网络请求框架
这里不作详解,具体的介绍可以看扔物线的这篇文章,对RxJava的入门者有很大的启发。其中也讲到了RxJava和Retrofit如何结合来实现更简洁的代码
四、RxJava代替EventBus进行数据传递:RxBus
注意:RxBus并不是一个库,而是一种模式,是使用了RxJava的思想来达到EventBus的数据传递效果。这篇文章把RxBus讲的比较详细。

五、使用combineLatest合并最近N个结点
例如:注册的时候所有输入信息(邮箱、密码、电话号码等)合法才点亮注册按钮。
  1. Observable<CharSequence> _emailChangeObservable = RxTextView.textChanges(_email).skip(1);
  2. Observable<CharSequence> _passwordChangeObservable = RxTextView.textChanges(_password).skip(1);
  3. Observable<CharSequence>   _numberChangeObservable = RxTextView.textChanges(_number).skip(1);
  4. Observable.combineLatest(_emailChangeObservable,
  5. _passwordChangeObservable,
  6. _numberChangeObservable,
  7. new Func3<CharSequence, CharSequence, CharSequence, Boolean>() {
  8. @Override
  9. public Boolean call(CharSequence newEmail,
  10. CharSequence newPassword,
  11. CharSequence newNumber) {
  12. Log.d("xiayong",newEmail+" "+newPassword+" "+newNumber);
  13. boolean emailValid = !isEmpty(newEmail) &&
  14. EMAIL_ADDRESS.matcher(newEmail).matches();
  15. if (!emailValid) {
  16. _email.setError("Invalid Email!");
  17. }
  18. boolean passValid = !isEmpty(newPassword) && newPassword.length() > 8;
  19. if (!passValid) {
  20. _password.setError("Invalid Password!");
  21. }
  22. boolean numValid = !isEmpty(newNumber);
  23. if (numValid) {
  24. int num = Integer.parseInt(newNumber.toString());
  25. numValid = num > 0 && num <= 100;
  26. }
  27. if (!numValid) {
  28. _number.setError("Invalid Number!");
  29. }
  30. return emailValid && passValid && numValid;
  31. }
  32. })//
  33. .subscribe(new Observer<Boolean>() {
  34. @Override
  35. public void onCompleted() {
  36. log.d("completed");
  37. }
  38. @Override
  39. public void onError(Throwable e) {
  40. log.d("Error");
  41. }
  42. @Override
  43. public void onNext(Boolean formValid) {
  44. _btnValidIndicator.setEnabled(formValid);
  45. }
  46. });


六、使用merge合并两个数据源。

例如一组数据来自网络,一组数据来自文件,需要合并两组数据一起展示。

  1. Observable.merge(getDataFromFile(), getDataFromNet())
  2. .observeOn(AndroidSchedulers.mainThread())
  3. .subscribe(new Subscriber<String>() {
  4. @Override
  5. public void onCompleted() {
  6. log.d("done loading all data");
  7. }
  8. @Override
  9. public void onError(Throwable e) {
  10. log.d("error");
  11. }
  12. @Override
  13. public void onNext(String data) {
  14. log.d("all merged data will pass here one by one!")
  15. });

七、使用concat和first做缓存

依次检查memory、disk和network中是否存在数据,任何一步一旦发现数据后面的操作都不执行。

  1. Observable<String> memory = Observable.create(new Observable.OnSubscribe<String>() {
  2. @Override
  3. public void call(Subscriber<? super String> subscriber) {
  4. if (memoryCache != null) {
  5. subscriber.onNext(memoryCache);
  6. } else {
  7. subscriber.onCompleted();
  8. }
  9. }
  10. });
  11. Observable<String> disk = Observable.create(new Observable.OnSubscribe<String>() {
  12. @Override
  13. public void call(Subscriber<? super String> subscriber) {
  14. String cachePref = rxPreferences.getString("cache").get();
  15. if (!TextUtils.isEmpty(cachePref)) {
  16. subscriber.onNext(cachePref);
  17. } else {
  18. subscriber.onCompleted();
  19. }
  20. }
  21. });
  22. Observable<String> network = Observable.just("network");
  23. //依次检查memory、disk、network
  24. Observable.concat(memory, disk, network)
  25. .first()
  26. .subscribeOn(Schedulers.newThread())
  27. .subscribe(s -> {
  28. memoryCache = "memory";
  29. System.out.println("--------------subscribe: " + s);
  30. });

八、使用timer做定时操作。当有“x秒后执行y操作”类似的需求的时候,想到使用timer

例如:2秒后输出日志“hello world”,然后结束。

  1. Observable.timer(2, TimeUnit.SECONDS)
  2. .subscribe(new Observer<Long>() {
  3. @Override
  4. public void onCompleted() {
  5. log.d ("completed");
  6. }
  7. @Override
  8. public void onError(Throwable e) {
  9. log.e("error");
  10. }
  11. @Override
  12. public void onNext(Long number) {
  13. log.d ("hello world");
  14. }
  15. });

九、使用interval做周期性操作。当有“每隔xx秒后执行yy操作”类似的需求的时候,想到使用interval

例如:每隔2秒输出日志“helloworld”。

  1. Observable.interval(2, TimeUnit.SECONDS)
  2. .subscribe(new Observer<Long>() {
  3. @Override
  4. public void onCompleted() {
  5. log.d ("completed");
  6. }
  7. @Override
  8. public void onError(Throwable e) {
  9. log.e("error");
  10. }
  11. @Override
  12. public void onNext(Long number) {
  13. log.d ("hello world");
  14. }
  15. });

十、使用throttleFirst防止按钮重复点击

ps:debounce也能达到同样的效果

  1. RxView.clicks(button)
  2. .throttleFirst(1, TimeUnit.SECONDS)
  3. .subscribe(new Observer<Object>() {
  4. @Override
  5. public void onCompleted() {
  6. log.d ("completed");
  7. }
  8. @Override
  9. public void onError(Throwable e) {
  10. log.e("error");
  11. }
  12. @Override
  13. public void onNext(Object o) {
  14. log.d("button clicked");
  15. }
  16. });

十一、使用schedulePeriodically做轮询请求

  1. Observable.create(new Observable.OnSubscribe<String>() {
  2. @Override
  3. public void call(final Subscriber<? super String> observer) {
  4. Schedulers.newThread().createWorker()
  5. .schedulePeriodically(new Action0() {
  6. @Override
  7. public void call() {
  8. observer.onNext(doNetworkCallAndGetStringResult());
  9. }
  10. }, INITIAL_DELAY, POLLING_INTERVAL, TimeUnit.MILLISECONDS);
  11. }
  12. }).subscribe(new Action1<String>() {
  13. @Override
  14. public void call(String s) {
  15. log.d("polling….”));
  16. }
  17. })

十二、RxJava进行数组、list的遍历

  1. String[] names = {"Tom", "Lily", "Alisa", "Sheldon", "Bill"};
  2. Observable
  3. .from(names)
  4. .subscribe(new Action1<String>() {
  5. @Override
  6. public void call(String name) {
  7. log.d(name);
  8. }
  9. });


十三、解决嵌套回调(callback hell)问题

  1. NetworkService.getToken("username", "password")
  2. .flatMap(s -> NetworkService.getMessage(s))
  3. .subscribe(s -> {
  4. System.out.println("message: " + s);
  5. })

十四、响应式的界面

比如勾选了某个checkbox,自动更新对应的preference

  1. SharedPreferences preferences = PreferenceManager.getDefaultSharedPreferences(this);
  2. RxSharedPreferences rxPreferences = RxSharedPreferences.create(preferences);
  3. Preference<Boolean> checked = rxPreferences.getBoolean("checked", true);
  4. CheckBox checkBox = (CheckBox) findViewById(R.id.cb_test);
  5. RxCompoundButton.checkedChanges(checkBox)
  6. .subscribe(checked.asAction());

最后,由于个人能力有限,文章难免有疏漏之处,如果您有任何疑议,请让我知道,谢谢!本文所有的例子已经上传到github上

致谢:这篇文章的绝大多数例子是从这里总结的,还有部分例子来自这里。对作者的无私贡献表示感谢!

RxJava使用场景小结的更多相关文章

  1. 可能是东半球最全的RxJava使用场景小结

    一.Scheduler线程切换 这样的场景常常会在"后台线程取数据,主线程展示"的模式中看见 Observable.just(1, 2, 3, 4) .subscribeOn(Sc ...

  2. Data URI 应用场景小结

    Data URI scheme 在前端开发中是个常用的技术,通常会在 CSS 设置背景图中用到.比如在 Google 的首页就有用到: Data URI scheme 简称 Data URI,经常会被 ...

  3. RxJava 复杂场景 Schedulers调度

    参考: https://blog.piasy.com/2016/10/14/Complex-RxJava-2-scheduler/ 以Zip为例,学习Schedulers的线程调度 要求: * cre ...

  4. Nginx也应用场景小结

    Nginx简介    Nginx一是一款轻量级的.高性能的HTTP和反向代理服务器, 具有很高的稳定性和支持热部署.模块扩展也很容易.当遇到访问的峰值,或者有人恶意发起慢速连接时,也很可能会导致服务器 ...

  5. Nginx---应用场景小结

    Nginx介绍   Nginx一是一款轻量级的.高性能的HTTP.反向代理服务器,具有很高的稳定性.支持热部署.模块扩展也非常容易.Nginx采取了分阶段资源分配技术,处理静态文件和无缓存的反向代理加 ...

  6. 【腾讯Bugly干货分享】基于RxJava的一种MVP实现

    本文来自于腾讯bugly开发者社区,非经作者同意,请勿转载,原文地址:http://dev.qq.com/topic/57bfef673c1174283d60bac0 Dev Club 是一个交流移动 ...

  7. 关于 RxJava 技术介绍

    Awesome-RxJava RxJava resources Blog 给 Android 开发者的 RxJava 详解 -强烈推荐 扔物线的文章 讲解非常详细 NotRxJava懒人专用指南 -这 ...

  8. 开发 Material Design+RxJava+Retrofit+MVP App 参考资料

    前言     在开发一个基于 Material Design+RxJava+Retrofit+MVP 框架的 App 过程中学习的资料整理 —— 由G军仔分享 这里记录了我开发 大象 项目时,所学习的 ...

  9. 我的Android进阶之旅------>RxJava学习资料汇总

    在响应式编程中,应该牢记以下两点: everything is a stream(一切皆流) don't break the chain(不要打断链式结构) 记住,可观测序列就像一条河,它们是流动的. ...

随机推荐

  1. nesC 语言参考手册

    1 简介 nesC 是对 C 的扩展 ,它基于体现 TinyOS 的结构化概念和执行模型而设计. TinyOS 是为传感器网络节点而设计的一个事件驱动的操作系统,传感器网络节点拥有非常有限的资源 ( ...

  2. 我是如何学习NodeJs

    实际上在开始的时候我已经对NodeJS有了一定的了解. 比如我知道它是居于Javascript语言的服务器端web Server,比如我知道它的优势在于它的性能,而造成性能优异的原因在于高效的V8引擎 ...

  3. 【UVA10603】Fill (构图+最短路)

    题目: Sample Input22 3 4 296 97 199 62Sample Output2 29859 62 题意: 有三个杯子它们的容量分别是a,b,c, 并且初始状态下第一个和第二个是空 ...

  4. #ifdef __cplusplus extern c #endif 的作用

    #ifdef __cplusplus // C++编译环境中才会定义__cplusplus (plus就是"+"的意思) extern "C" { // 告诉编 ...

  5. 如何在jasperreport自动生成序号

    在导出报表时,有时候我们需要显示序号,有两种方法: 1.就是再加一个字段,就是说将序号也当做是要导出的字段来处理,然后用程序给这个字段赋值,这方面有点傻,就不说了. 2.利用jasperreport提 ...

  6. 什么是系统,什么是算法 -- lemon OA 系统学习总结

    一.对于模块划分的理解 对于一个大型(这里还只是一个中型系统)系统来说,代码的编写不是一日而就的,层次的明细也不是一眼就能看清楚的. 在这种情况下,需要想好,最好是由上而下地想好. 能够模块式地划分最 ...

  7. org.springframework.web.servlet.view

    view包下面的类和接口 description:提供view和viewResolver的标准实现,也提供一些抽象基类.Spring MVC已经提供了 JSPs, Velocity, XSLT等视图的 ...

  8. eclipse中 com.sun.image.codec.jpeg.JPEGCodec 无法编译通过问题

    在Eclipse中处理图片,需要引入两个包:import com.sun.image.codec.jpeg.JPEGCodec;import com.sun.image.codec.jpeg.JPEG ...

  9. codeforces --- 115A

    A. Party time limit per test 3 seconds memory limit per test 256 megabytes input standard input outp ...

  10. Longest Consecutive Sequence——Leetcode

    Given an unsorted array of integers, find the length of the longest consecutive elements sequence. F ...