博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
【译】Android Architecture - ViewModel 与 View 的通信
阅读量:6701 次
发布时间:2019-06-25

本文共 4608 字,大约阅读时间需要 15 分钟。

前言

本文翻译自【】,介绍了 MVVM 架构中 VM 与 V 的通信。感谢作者 。水平有限,欢迎指正讨论。 自从 Google 在去年 I/O 大会发布 以来,MVVM 架构已经成为一种趋势。很多之前熟练于 MVP 架构的开发者,现在也慢慢开始接受并使用 MVVM 架构了。相比于 Presenter,使用 有以下好处:减少很多样板代码,配置变更时自动恢复数据,可以轻松地在多个 Fragment 之间共享数据。然而,ViewModel 与 View 之间的通信变得更加困难。

正文

痛点

以一个用户信息修改界面为例,在请求服务器之前,必须先校验用户数据,而 Presenter 或 ViewModel 的职责就是显示和取消 Loading,以及将校验或服务器的返回结果展示到界面上。此外,如果一个 Dialog 正在显示,当配置变更后也应该恢复 Dialog。

Presenter 和 ViewModel 不应持有 View 的引用。

在 MVP 架构中,我们经常需要定义一些契约类接口(Contract),View 实现 Contract.View 接口,Presenter 实现 Contract.Presenter 接口,在 Presenter 中不持有 Activity/Fragment 的引用,只持有 View 实例,这样可以方便地调用 View 接口暴露的方法。 例如 EditProfileContract.kt

interface EditProfileContract {    interface view {        fun setProgress(show: Boolean)        fun showEmptyFirstNameError()        fun showEmptyLastNameError()    }    interface presenter {        fun saveProfile(firstName: String, lastName: String, bio: String, email: String, city: City, gender: String)    }}复制代码

但是,在 MVVM 架构中,ViewModel 不再持有 View 的引用,而是通过 或 向 View 层暴露数据。一旦 View 订阅了 ViewModel,它就开始接收数据更新。这看似很完美,但当 ViewModel 想要更新 View 状态,比如显示和取消 Loading,将数据校验或服务器结果反馈到 UI 界面上,会变得非常困难。

解决方案

ViewModel 中的 LiveData 或 越少越好。因此我们最好找到一种方法,可以封装需要传递给 View 层的数据和信息。在多数情况下,ViewModel 需要向 View 层暴露以下三种数据:

  • Data
  • Status
  • State 下面将依次介绍。

Data

Data -- 就是需要在 View 上展示的内容,比如用户信息的 User 实体类,或社交 Feed 流中的列表项。

val user = MutableLiveData
()val feeds = MutableLiveData
>()复制代码

Status

Status -- 可以是任何仅需传递一次的信息,如校验错误,网络异常,或者服务器错误。 :

enum class Status {    SUCCESS,    ERROR,    NO_NETWORK,    EMPTY_FIRST_NAME,    EMPTY_LAST_NAME,    EMPTY_CITY,    INVALID_URI}复制代码

LiveData 没有提供任何开箱即用的方法,但在 Google 的官方示例中,有一个 的实现,可以解决这个问题。

一个生命周期感知的被观察者,仅在订阅后发送新的更新,常用于导航和 Snackbar 消息等事件。 这可以避免一些常见问题:在配置变更(如屏幕旋转)期间,如果观察者处于活动动态,SingleLiveEvent 将会发送更新事件。 它继承于 MutableLiveData,是一个被观察者,即使对外暴露了 SingleLiveEvent#setValue()SingleLiveEvent#call() 方法, 注意:只有一个观察者会受到更新通知。

新建一个 SingleLiveEvent 用来向 View 层暴露 Status 数据。 :

private val status = SingleLiveEvent
()fun getStatus(): LiveData
{ return status}fun handleImage(intent: Intent?) { intent?.data?.let { avatar.value = it.toString() } ?: run { status.value = Status.INVALID_URI }}复制代码

View 只关心 Status 数据,并根据不同的状态或错误执行对应的逻辑。如下实例,我们能很方便地根据每个错误显示不同的 Toast 或 Snackbar。 :

viewModel.getStatus().observe(this, Observer { handleStatus(it) })private fun handleStatus(status: Status?) {    when (status) {        Status.EMPTY_FIRST_NAME -> Toast.makeText(activity, "Please enter your first name!", Toast.LENGTH_SHORT).show()        Status.EMPTY_LAST_NAME -> Toast.makeText(activity, "Please enter your last name", Toast.LENGTH_SHORT).show()        Status.EMPTY_CITY -> Toast.makeText(activity, "Please choose your home city", Toast.LENGTH_SHORT).show()        Status.INVALID_URI -> Toast.makeText(activity, "Unable to load the photo", Toast.LENGTH_SHORT).show()        Status.SUCCESS -> {            startActivity(HomeFragment.newIntent(activity))            activity.finish()        }        else -> Toast.makeText(activity, "Something went wrong, please try again!", Toast.LENGTH_SHORT).show()    }}复制代码

State

State -- 即 UI 状态,比如加载进度条和 Dialog 等,每次开始订阅 ViewModel 的数据时,ViewModel 应该把这些 UI 状态通知给 View 层。一种简单的做法是,我们可以创建一个数据类来保存这些状态。 :

data class EditProfileState(    var isProgressIndicatorShown: Boolean = false,    var isCityDialogShown: Boolean = false,    var isGenderDialogShown: Boolean = false)复制代码

然后在 ViewModel 中创建一个 MutableLiveData,用来包装这个 EditProfileState。由于 ViewModel 只会暴露 LiveData 给 View 层,因此我们应该提供 setter 方法,便于 View 更新此状态。 :

private val state = MutableLiveData
()fun getState(): LiveData
{ return state}fun setProgressIndicator(isProgressIndicatorShown: Boolean) { state.value?.isProgressIndicatorShown = isProgressIndicatorShown}fun setCityDialogState(isCityDialogShown: Boolean) { state.value?.isCityDialogShown = isCityDialogShown}fun setGenderDialogState(isGenderDialogShown: Boolean) { state.value?.isGenderDialogShown = isGenderDialogShown}复制代码

最后,根据上面的 State 状态数据,决定 Dialog 的显示和取消。 :

viewModel.getState().observe(this, Observer { handleState(it) })private fun handleState(state: EditProfileState?) {    if (state?.isCityDialogShown == true) {        showCitySelectionDialog()        return    }    if (state?.isGenderDialogShown == true) {        showGenderSelectionDialog()        return    }}复制代码

总结

封装诸如 loading 状态,UI 状态或服务器错误等信息,可以让 ViewModel 保持干净简洁。对我来说,StatusState 是一种好的解决方案。

评论中的问题

  • 关于 enum 的使用: -结论:In fact, if you use enums, I don't care. Go ahead

  • 关于使用 :

    • 可以使用 DataBinding 解决 VM 和 V 的通信。

参考

联系

我是 xiaobailong24,您可以通过以下平台找到我:

  • Github:
  • 简书:
  • 掘金:

转载地址:http://qcwlo.baihongyu.com/

你可能感兴趣的文章
我的友情链接
查看>>
Spray.io搭建Rest服务
查看>>
探索C++对象模型(二)
查看>>
内核模式和用户模式
查看>>
SSH 整合框架(自整理)
查看>>
学习ARM嵌入式linux的一些建议
查看>>
java.lang.NoClassDefFoundError解决方案
查看>>
textView限制字数(超简单,不走弯路)(解决联想输入及iOS7崩溃等问题)
查看>>
shell实例
查看>>
我的友情链接
查看>>
java中四种进制的转换
查看>>
git多个远程仓库
查看>>
Linux之命令
查看>>
Android 6.0 特性
查看>>
shell 脚本作业
查看>>
程序员老司机都要错的 Python 陷阱与缺陷列表
查看>>
《netty入门与实战》笔记-06:心跳与空闲检测
查看>>
使用javascript开发的视差滚动效果的云彩 极客标签 - 做最棒的极客知识分享平台...
查看>>
SSM整合框架
查看>>
【安全牛学习笔记】CONTROL FRAME
查看>>