MVVM_Android-CleanArchitecture

原文发表于: Rocko's blog - MVVM_Android-CleanArchitecture

前言

"Architecture is About Intent, not Frameworks" - Robert C. Martin (Uncle Bob)

Uncle Bob 的这句话套在 MVVM 上也是适用的, MVVM 也仅仅是架构模式(Architectural pattern),其有一套自己的理论概念(pattern)而不是规定的具体实现(或 Frameworks)。早之前在知乎上相关问题的回答(android UI设计MVVM设计模式讨论?)中也简单提到过 MVVM 了,M-V-X 的关系如上图,那么这一次博主把 Fernando Cejas(android10)Android-CleanArchitecture 项目中的 MVP 实现重构成了用 MVVM 来实现。所以看这篇文章最好是先搞清了 Fernando Cejas(android10)Android-CleanArchitecture sample app 和对应的两篇文章(见参考)。整个历程也算比较愉快,没什么不良反应,这篇文章理所当然会重点说说 MVVM 的实现、 Data Binding 等相关的东西。那为什么拥抱 MVVM 呢。当然是 Google 推出官方的 data binding 啦,下一次的 Android MVVM 热潮应该就是 data binding 放出正式版了。

分层架构与 M-V-X

首先还是先来说说 分层架构MVC or M-V-X 之间的关系。分层架构是一种常见的软件应用架构,在 Java 程序中可以算是一种应用标准了,通常又叫 N 层架构,而最常见的是 3 层架构,它包含如下 3 层:

  • 展示层(Presentation tier),也称为 UI 层,也就是程序的界面部分。

  • 业务层(business logic(domain) tier), 业务层,是最为核心的一层。

  • 持久层(Data tier),数据持久层。

3 层架构是存在物理上分层概念的,从上往下即展示层、业务层、持久层,也从上往下由上一层依赖下一层。不同层之间也是 高内聚低耦合 的体现,层内高内聚,层间低耦合, 是层内具体工作的高度抽象。低耦合则是依赖倒转原则体现出来,高层依赖于下层的抽象而不是具体。

接下来先说 M-V-X 的鼻祖 MVC,Model–View–Controller (MVC) is a software architectural pattern for implementing user interfaces.,所以 MVC 模式是为用户界面设计的,在 3 层架构中,MVC 是属于展现层的部分,所以 MVVM 作为 MVC 的演进在与分层架构的关系上也是一样。经常会看到 3 层架构与 M-V-X 混为一谈的内容,这是不正确的,虽然都是 3 部分的内容但是不能 简单地 把两者的每一部分对应起来,我们应该理解为,在分层架构中 M-V-X 是在展示层(Presentation tier)的应用。这些软件工程理论的东西就这样了,都是大师们留下来的东西,不要随便套上自己的概念。

Android-CleanArchitecture

Fernando Cejas(android10)Android-CleanArchitecture 项目中也是采用典型的 3 层架构,其中 Presentation 展现层采用了 MVP 模式,如果还未了解过 MVP,可以看看我之前写的文章:Android中的MVP。不过 Google 推出官方的 data binding 之后我觉得基本可以不用采用 MVP 了,在 M-V-X 中, MVP 与 MVVM 算是比较接近的了但 MVP 中的一堆 View 接口也是让人头疼的,而拥有 data binding 的 MVVM 则解决了这个问题,所以请大胆拥抱 MVVM。So,下面几点是当中除 MVVM 外涉及到的东西,MVVM 放到下一节再讲。

Dagger

<i class="fa fa-eye-slash"></i> 自带 Dagger 信仰光环者障眼之术开启<i class="fa fa-eye-slash"></i>, 哈哈,你们看不到接下来的这句话了。。我现在也是持不赞成 di 的观点的人(在 Android 中、、、),可以看看这场撕逼:依赖注入是否值得?。结果就是我把原项目的依赖注入模块去掉了,对于测试中的类中的成员变量来说本来就是 Mock 抽象接口,那就直接对接口或抽象类直接 Mock 操作就可,毕竟依赖注入的解耦依然是取决于需要注入的对象的抽象,维护依赖注入模块(Module)也是负担,测试代码中又要多写一套注入控制的 Dagger Module 代码。。

RxJava、RxAndroid

先说 AsyncTask,对其已经不再想吐槽,这么重要的异步实现,版本间(Android Api)代码改来改去,又顺序又无序、又单线程执行又并发执行、内存泄露、、、。所以对于采用 RxJava 即使不采用函数响应式编程的大概念,用它来替换 AsyncTask 和 Thread + Handler 也是推荐的。此外使用 Rx 后也不需要事件总线的框架了,对于回调监听直接在数据操作的 Observable 上注册观察者即可,相对于事件总线来说是更精准的(单线)的监听。而事件总线的话则是更加松耦合的,出错的话会更加难排查,这里就不再展开了。

Lambda

个人、团队喜好,代码简洁了很多但是代码的逻辑比较不好直观理解了,原项目也只有 3 处代码用到,故而去掉了。

此外,对于领域驱动设计(DDD)中 Repository,原项目中把其实现放到了 data 层去实现(接口),造成 data 层会依赖其上一层(domain 业务层),作为分层架构个人认为不合适所以重新把它调整了。根据 DDD,个人认为 Repository 它的存在让领域层(domain 业务)感觉不到数据访问层的存在,它提供一个类似集合的接口提供给领域层进行领域对象的访问。Repository 是仓库管理员,领域层需要什么东西只需告诉仓库管理员,由仓库管理员把东西拿给它,并不需要知道东西实际放在哪。

MVVM

理论简述

按照常理,先来说基本概念:

  • Model,domain model(领域模型)或是数据层代表的数据模型,也可以理解为用户界面需要显示数据的抽象(数据)

  • View, 应用的界面

  • ViewModel,binder 所在之处,是 View 的抽象,对外暴露出公共属性和命令,是 View 与 Model 的(绑定)连接器

此外还有必不可少的一部分:Binder,Android 中也就是 Data binding 了,提供 View 与 Model 的绑定功能。下面是结构图:

MVVM结构图

Android 中实现

目前 Android 的 data binding 还是 beta,还只是 one-way 单向绑定,功能上还有所欠缺、控制性也还不强,但是把它写出来还是没问题的。对于 Activity、Fragment 而言仅仅是作为 Java View 看待,与 XML 对应,所以里面只有 View 的展现逻辑,此外没有其它代码。一个 Activity 或 Fragment(一般都 with XML) 对应一个 ViewModel,对于一个基础 View(XML)可以通过继承对应的 ViewModel 实现重用,本文的代码也有体现。对于 Activity 和 Fragment 的View 状态保存恢复也通过 ViewModel 处理。因为 binding 的入口在 Activity 或 Fragment 中,所以为了方便写个基类处理 ViewModel 和 Binding 的初始化,然后在 对应的 XML 里加上 ViewModel 的 variable, XML 里不再有其它数据对象的 variable。

public abstract class BaseActivity<VM extends ViewModel, B extends ViewDataBinding> extends Activity {

  private VM viewModel;
  private B binding;

  public void setViewModel(@NonNull VM viewModel) {
    this.viewModel = viewModel;
  }

  public VM getViewModel() {
    if (viewModel == null) {
      throw new NullPointerException("You should setViewModel first!");
    }
    return viewModel;
  }

  public void setBinding(@NonNull B binding) {
    this.binding = binding;
  }

  public B getBinding() {
    if (binding == null) {
      throw new NullPointerException("You should setBinding first!");
    }
    return binding;
  }

}

ViewModel 中通过 ObservableField 来达到细粒度的控制,绑定操作都放在 ViewModel 里,然后 ViewModel 里可以有多个 domain 中的 Interator(UseCase) 来得到 View 需要渲染的数据 Model。对于 ObservableField 的绑定操作和命令操作(Command)都是暴露的,也易于测试。binding 现在缺少手动在 Java 代码中注册通知事件的功能,比如有些 model 的渲染必须通过 Java 代码来操作的话就需要了,在 Activity(Java View) 中通过向 Binding 注册通知回调,而目前只能在 XML 中知道,当然目前也可以自己实现,方法也有多种:接口回调、EventBus、RxBus。。

架构图

除了 Persenter 改成 ViewModel 的逻辑、Repository 的抽象和具体都在 domain 外,其他部分基本一致,采用的测试也一致。

  • Clean Architecture:
Clean Architecture
  • ****MVVM_Clean-Architecture tier:****


    MVVM_Clean-Architecture 分层结构
  • MVVM_Clean-Architecture put all:

    MVVM_Clean-Architecture put all 应用在一起

Refactor

Talk is cheap. Show you the code. ↓↓↓
MVVM_Android-CleanArchitecture

需要注意的是 include 标签的 XML 节点中要使用到根节点中 data 标签里设置的 viewModel variable 的话需要这样设置;

  <include
      layout="@layout/view_retry"
      bind:viewModel="@{viewModel}"/>

抽象类 ViewModel 中设置了 @Command 和 @BindView 注解,只起到清晰提醒作用。

具体重构更改可以查看 commit 记录:MVVM_Android-CleanArchitecture commits。可以看到 Activity 和 Fragment 的代码是很清爽的,比 MVP 更清爽,因为 View 的数据渲染操作交给 binder 去处理了。对 Activity 或其对应界面进行 UI 测试的话,Mock 出 model 代表的数据然后传递给 ViewModel 中 @BindView 暴露出的方法,然后检验视图对数据的正确显示就行了,也就是 View 对 Model 做了正确的渲染。

参考

企业应用架构模式
Architecting Android…The clean way? (Android-CleanArchitecture 的文章,译文略)
Architecting Android…The evolution (Android-CleanArchitecture 的文章,译文略)
Approaching Android with MVVM
ANDROID DATABINDING: GOODBYE PRESENTER, HELLO VIEWMODEL!

END

本文源码:MVVM_Android-CleanArchitecture or Rocko-Android-Demo(-MVVM_Android-CleanArchitecture)

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 194,524评论 5 460
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 81,869评论 2 371
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 141,813评论 0 320
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 52,210评论 1 263
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 61,085评论 4 355
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 46,117评论 1 272
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 36,533评论 3 381
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 35,219评论 0 253
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 39,487评论 1 290
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 34,582评论 2 309
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 36,362评论 1 326
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,218评论 3 312
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 37,589评论 3 299
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 28,899评论 0 17
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,176评论 1 250
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 41,503评论 2 341
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 40,707评论 2 335

推荐阅读更多精彩内容