http://jcodecraeer.com/a/anzhuokaifa/androidkaifa/2016/0131/3930.html

最近 RxAndroid 、MVP、MVVM 一直是 Android 程序猿茶余饭后的谈资,于是我也抱着凑热闹的态度试试了试水。这里就谈谈试水后的感受

什么是 RxAndroid ?

要说什么是 RxAndroid ,得从 RxJava 说起。RxJava 在 GitHub 主页上的自我介绍是 “a library for composing asynchronous and event-based programs using observable sequences for the Java VM”(一个在 Java VM 上使用可观测的序列来组成异步的、基于事件的程序的库)。这就是 RxJava ,概括得非常精准。

RxJava 的本质可以压缩为异步这一个词。说到根上,它就是一个实现异步操作的库,而别的定语都是基于这之上的。

而RxAndroid是RxJava的一个针对Android平台的扩展,主要用于 Android 开发。

什么是 Retrofit ?

Retrofit 是一套 RESTful 架构的 Android(Java) 客户端实现,基于注解,提供 JSON to POJO(Plain Ordinary Java Object ,简单 Java 对象),POJO to JSON,网络请求(POST,GET, PUT,DELETE 等)封装。

既然只是一个网络请求封装库,现在已经有了那么多的大家已经耳熟能详的网络请求封装库了,为什么还要介绍它呢,原因在于 Retrofit 是一套注解形的网络请求封装库,让我们的代码结构更给为清晰。它可以直接解析JSON数据变成JAVA对象,甚至支持回调操作,处理不同的结果。主要是 Retrofit 能很好的与 RxAndroid 配合使用。

想更详细的了解 Retrofit,可以查看官方文档

什么是 MVVP ?

MVC(Model-View-Controller)和 MVP(Model-View-Presenter)是最常见的软件架构之一,业界有着广泛应用,大家一定不陌生。但知道什么事 MVVP 的就不多了,它本身很容易理解,但是要讲清楚,这几个架构的区别就不容易了。

MVVM(Model-View-ViewModel),它采用双向绑定(data-binding):View的变动,自动反映在 ViewModel,反之亦然。Angular 和 Ember 都采用这种模式。

而 Google 新推出的 Databinding 正是采用了这种模式。

RxAndroid + Retrofit + Databinding

上面已经分别介绍了 RxAndroid、Retrofit、Databinding ,想必大家也有了个初步的认识,那我们就看看 RxAndroid + Retrofit + Databinding 产生的“化学反应”。

  1. private void initActionBar() {
  2. setSupportActionBar(getBinding().toolbar);
  3. DrawerLayout drawer = getBinding().drawLayout;
  4. ActionBarDrawerToggle toggle = new ActionBarDrawerToggle(this, drawer, getBinding().toolbar, R.string.navigation_drawer_open, R.string.navigation_drawer_close);
  5. drawer.setDrawerListener(toggle);
  6. toggle.syncState();
  7. getBinding().navigationView.setNavigationItemSelectedListener(this);
  8. }

代码中不再充斥着 findViewById 这样的代码了,将 etContentView() 换成下面的方法。

  1. this.mBinding = DataBindingUtil.setContentView(context, layout_id);

系统会将我们的 layout 和 data 进行绑定并返回 bind 对象,bind.* 或者 bind.set 方法来取得控件或修改值。当然还有其它的方法,但是你此时再使用 findViewById() 方法不再有效了。

  1. public interface NewsApi {
  2. /**
  3. * 根据 ID 请求新闻列表
  4. * @param id
  5. * @return
  6. */
  7. @Headers("apikey: 2c61a1cd1f64216e92f7da1603697bf7")
  8. @GET(ApiConst.NEWS) Observable<News.NewsData> queryNewsByID(@Query("channelId") String id, @Query("page") int page);
  9. /**
  10. * 根据 ChannelName (标题)请求新闻列表
  11. * @param title
  12. * @return
  13. */
  14. @Headers("apikey: 2c61a1cd1f64216e92f7da1603697bf7")
  15. @GET(ApiConst.NEWS) Observable<News.NewsData> queryNewsByCName(@Query("channelName") String title, @Query("page") int page);
  16. /**
  17. * 根据 title (标题)请求新闻列表
  18. * @param title
  19. * @return
  20. */
  21. @Headers("apikey: 2c61a1cd1f64216e92f7da1603697bf7")
  22. @GET(ApiConst.NEWS) Observable<News.NewsData> queryNewsByTitle(@Query("title") String title, @Query("page") int page);
  23. }

  1. private void initObservables() {
  2. Observable.Transformer<List<News>, List<News>> networkingIndicator = RxNetworking.bindRefreshing(getBinding().refresher);
  3. observableRefresherNewsData = Observable.defer(() -> mNewApi.queryNewsByCName(getArguments().getString(BUNDLE_NAME), 1))
  4. .doOnUnsubscribe(() -> this.unsubcribe("observableNewsData"))
  5. .flatMap(data -> Observable.just(data.contentlist))
  6. .flatMap(list -> getApp().getDB().putList(Const.DB_NEWS_NAME, list, News.class))
  7. .subscribeOn(Schedulers.io())
  8. .observeOn(AndroidSchedulers.mainThread())
  9. .compose(networkingIndicator);
  10. observableLoadMoreNewsData = Observable.defer(() -> mNewApi.queryNewsByCName(getArguments().getString(BUNDLE_NAME), mCurrPage + 1))
  11. .doOnUnsubscribe(() -> this.unsubcribe("observableNewsData"))
  12. .map(data -> {
  13. mCurrPage = data.currentPage;
  14. return data.contentlist;
  15. })
  16. .subscribeOn(Schedulers.io())
  17. .observeOn(AndroidSchedulers.mainThread())
  18. .compose(networkingIndicator);
  19. // 刷新/加载更多
  20. RxSwipeRefreshLayout.refreshes(getBinding().refresher)
  21. .doOnUnsubscribe(() -> this.unsubcribe("SwipeRefreshLayout"))
  22. .flatMap(avoid -> observableRefresherNewsData)
  23. .compose(bindToLifecycle())
  24. .subscribe(RxList.prependTo(mNews, getBinding().content), this::showError);
  25. RxEndlessRecyclerView.reachesEnd(getBinding().content)
  26. .doOnUnsubscribe(() -> this.unsubcribe("Recycler"))
  27. .flatMap(avoid -> observableLoadMoreNewsData)
  28. .compose(bindToLifecycle())
  29. .subscribe(RxList.appendTo(mNews), this::showError);
  30. // 首次进入手动加载
  31. observableRefresherNewsData
  32. .map(list -> {
  33. mNews.clear();
  34. return list;
  35. })
  36. .compose(bindToLifecycle())
  37. .subscribe(RxList.prependTo(mNews, getBinding().content), this::showError);
  38. }

上面代码是使用 Retrofit 以 Get 形式从服务器中获取对应的新闻数据,大家可以看到代码的逻辑非常清晰,代码也很简洁(这里使用了 lambda 表达式,不使用的话,代码会长些,但是逻辑依然清晰),如果是按以前的写法的话,我们的代码会比这复杂的多,还涉及到复杂的线程之间的通信。而通过 RxJava ,我们只需要简单的使用 subscribeOn(Schedulers.io()) 和 observeOn(AndroidSchedulers.mainThread()) 就可以完成 IO 线程和 UI 线程的切换。

帅的简直不敢相信,原来还可以这样玩。

总结

优点:

  1. 代码逻辑更多加清晰。

  2. 线程之间的切换更加方便、自如。

  3. 代码可扩展性高,便于维护。

  4. 不再为 findViewById() 方法而烦,为 Activity 减负,整体结构更加清晰。

缺点:

  1. 代码出错时,由于 RxJava 的原因,将不太容易找到具体出错位置。

  2. 由于 RxJava 结构问题,部分需要捕捉的错误可能被 RxJava 消化掉。

  3. Databinding 在部分情况使用不太如意,如 include 进来的 layout 里对应的 id 不会被关联起来。

  4. 需要一定的学习成本(当然这不是问题)。

广告

这里打个小广告,介绍下我最近开发的几个小应用

大家多支持下,如果下载达到 1000 的话,我会将其中一两个项目开源出来的哦。

扩展阅读

  1. RxJava / RxAndroid

  2. Retrofit:

  3. Databinding

浅谈 RxAndroid + Retrofit + Databinding的更多相关文章

  1. Android开发-浅谈架构(二)

    写在前面的话 我记得有一期罗胖的<罗辑思维>中他提到 我们在这个碎片化 充满焦虑的时代该怎么学习--用30%的时间 了解70%该领域的知识然后迅速转移芳草鲜美的地方 像游牧民族那样.原话应 ...

  2. [转帖]浅谈响应式编程(Reactive Programming)

    浅谈响应式编程(Reactive Programming) https://www.jianshu.com/p/1765f658200a 例子写的非常好呢. 0.9312018.02.14 21:22 ...

  3. 浅谈 Fragment 生命周期

    版权声明:本文为博主原创文章,未经博主允许不得转载. 微博:厉圣杰 源码:AndroidDemo/Fragment 文中如有纰漏,欢迎大家留言指出. Fragment 是在 Android 3.0 中 ...

  4. 浅谈 LayoutInflater

    浅谈 LayoutInflater 版权声明:本文为博主原创文章,未经博主允许不得转载. 微博:厉圣杰 源码:AndroidDemo/View 文中如有纰漏,欢迎大家留言指出. 在 Android 的 ...

  5. 浅谈Java的throw与throws

    转载:http://blog.csdn.net/luoweifu/article/details/10721543 我进行了一些加工,不是本人原创但比原博主要更完善~ 浅谈Java异常 以前虽然知道一 ...

  6. 浅谈SQL注入风险 - 一个Login拿下Server

    前两天,带着学生们学习了简单的ASP.NET MVC,通过ADO.NET方式连接数据库,实现增删改查. 可能有一部分学生提前预习过,在我写登录SQL的时候,他们鄙视我说:“老师你这SQL有注入,随便都 ...

  7. 浅谈WebService的版本兼容性设计

    在现在大型的项目或者软件开发中,一般都会有很多种终端, PC端比如Winform.WebForm,移动端,比如各种Native客户端(iOS, Android, WP),Html5等,我们要满足以上所 ...

  8. 浅谈angular2+ionic2

    浅谈angular2+ionic2   前言: 不要用angular的语法去写angular2,有人说二者就像Java和JavaScript的区别.   1. 项目所用:angular2+ionic2 ...

  9. iOS开发之浅谈MVVM的架构设计与团队协作

    今天写这篇博客是想达到抛砖引玉的作用,想与大家交流一下思想,相互学习,博文中有不足之处还望大家批评指正.本篇博客的内容沿袭以往博客的风格,也是以干货为主,偶尔扯扯咸蛋(哈哈~不好好工作又开始发表博客啦 ...

随机推荐

  1. Mybatis插件原理分析(一)

    我们首先介绍一下Mybatis插件相关的几个类,并对源码进行了简单的分析. Mybatis插件相关的接口或类有:Intercept.InterceptChain.Plugin和Invocation,这 ...

  2. ROS_Kinetic_02 ROS Kinetic 迁移指南及中文wiki指南(Migration guide)

    ROS_Kinetic_02 ROS Kinetic 迁移指南(Migration guide) 对于ROS Kinetic Kame有些功能包已经更新改变,提供关于这些包的迁移注意或教程.主要针对于 ...

  3. 05_学生管理系统,xml读写,布局的综合应用

     最终要做的项目目标: 2.编写Android清单文件AndroidManifest.xml <?xml version="1.0" encoding="utf ...

  4. Leetcode_8_String to Integer

    本文是在学习中的总结,欢迎转载但请注明出处:http://blog.csdn.net/pistolove/article/details/41521063 Implement atoi to conv ...

  5. BLOCK/字面量(语法糖)OC——第六天

    1.//block ,块语法,实质是匿名函数,是对C语言中函数的扩充,扩展: //block  语法可以用来保存一段代码或者用来调用一段封装好的代码: //block  语法由于是C语言实现的,所以执 ...

  6. 数据的压缩存储与解压缩算法实现(C语言)

    在一些嵌入式的项目设计中,空间是相当宝贵的,因为一个CPU的存储是有限的,所以此时我们在保存数据的时候,喜欢来进行压缩保存,著名的有哈夫曼树算法,专门用来做压缩的算法,当然,本节我们不讨论这些稍微高级 ...

  7. Android 上滑上拉菜单SlidingDrawer 不全屏显示的方法

    这里来说一个已经被废弃的SlidingDrawer.. 他可以实现上拉,下拉的菜单. 但是有个问题就是上拉以后,是全屏显示的. 首先 写一个布局: <RelativeLayout xmlns:a ...

  8. window环境下搭建react native及相关插件

    可以先浏览一下中文翻译的开发文档具体了解一下关于React Native,想要查看官方文档可以点http://facebook.github.io/react-native/docs/getting- ...

  9. UML之时序图

            时序图,英文名曰:Sequence Diagram,也称顺序图和序列图,是一种行为图,她通过描述对象之间发送消息的时间顺序显示多个对象之间的动态协作.她可以表示用例的行为顺序,当执行一 ...

  10. Callable与Future

    本文可作为传智播客<张孝祥-Java多线程与并发库高级应用>的学习笔记. 在前面写的代码中,所有的任务执行也就执行了,run方法的返回值为空. 这一节我们说的Callable就是一个可以带 ...