Android Jetpack -- ViewModel 简析

ViewModel

不接受任何形式的转发,如有需要请联系作者,谢谢理解!

Android Jetpack 是一系列助力您更容易打造优秀 Android 应用的工具和组件。这些组件能帮助您遵循最佳实践、免除编写繁复的样板代码并简化复杂任务,从而使您可以专注于最核心的代码逻辑。 Jetpack 中的架构指南由 Android 开发中四个关键领域中的一系列代码库和工具提供支持。它们分别是基础、架构、行为和 UI。每个 Jetpack 组件均可单独采用,但它们依然可以流畅地协作。其中的 Lifecycle 库,可以有效避免内存泄漏和解决常见的 Android 生命周期难题,今天将介绍 Lifecycle 库中的 ViewModel 类 。

Architecture Component

介绍

简单来说,ViewModel 是用来保存应用 UI 数据的类,而且它会在配置变更后继续存在。它可以脱离 View 单纯做 Junit 的测试,更方便大家做单元测试。ViewModel 有这样的特性,但是它在实际项目能够有何表现,还需要看待各位是如何使用的。一般会和 Livedata、DataBinding 等其他组件进行组合使用,下面会介绍具体可以解决什么问题,和一般项目中会建议如何使用。在介绍的过程中,还会穿插一些架构的思想以及如何看源码分析源码的思路。

解决了什么问题

手机屏幕旋转是配置变更的一种,当旋转屏幕时, Activity 会被重新创建。如果数据没有被正确的保存和恢复,就有可能丢失。从而导致莫名其妙的 UI 错误,甚至应用崩溃。Demo 请点击这里

kyHxB9.gif
应用 UI 数据丢失

相反的, ViewModel 会在配置更改后继续存在,所以,如果将应用所有的 UI 数据保存在 ViewModel 中,而不是 Activity 中,这样就可以保证数据不会受到配置变更的影响了。

kybeHA.gif
数据保存在 ViewModel 中
ViewModel 生命周期

新手常见犯的错误是将很多业务变量,逻辑和数据都摆在 Activity 或 Fragment 中,这样的代码比较混乱和难以维护,这种开发模式违犯了单一责任的原则。而 ViewModel 可以有效地划分责任,它可以用来保存 Activity 的所有 UI 数据,然后 Activity 仅负责了解如何在屏幕上显示数据和接收用户互动,但是 Activity 不会处理这些互动所带来的业务逻辑(如:点击获取短信验证码按钮后的网络请求)。但不要将太多的逻辑处理,数据获取放到 ViewModel 类里面处理,它仅仅作为 UI 数据的保存!

简单使用

上面提到,不要将太多的逻辑处理、数据存储获取放到 ViewModel 类里面处理,它仅仅作为 UI 数据的保存!数据存储获取可以创建 Repository 类, UI 数据处理可以使用 Presenter 类。Demo 请点击这里

AkWpKH.png

ViewModel 的使用方式如下:

implementation "androidx.lifecycle:lifecycle-viewmodel-ktx:$lifecycle_version"
//继承系统 ViewModel 类
class UserProfileViewModel:ViewModel(){
    //将跟 UI 界面有关的数据放至 UserProfileViewModel 中
    val user = User(name="",company="")
}
//在 Activity 中实例化 UserProfileViewModel 变量 
override fun onCreate(saveInstanceState:Bundle?){
    //Setup Activity
    val userViewModel =  ViewModelProviders.of(this).get(UserProfileViewModel::class.java)
    userViewModel.user.name = "Edgar Ng"
}

实际应用

ViewModel 一般会搭载其他组件(如:LiveData、DataBinding 等)一同使用,本系列文章会专门出一篇文章就 ViewModel 及其他组件搭配使用进行说明讲解,请点击此处

本章也会通过购买VIP会员这个业务功能点来简述实际项目中会如何使用 ViewModel 。具体代码链接请点击这里

k4TX7T.png
k4odiQ.png

k5FfHO.jpg

k5F4ED.jpg

kjtFoj.gif

该 Demo 总得来说分为三层:展示层(Presentation Layer)、域层(Domain Layer)、数据层(Data Layer),如下图所示。

  • 展示层:View,ViewModel
  • 域层:use cases
  • 数据层:repositories

展示层无法直接与数据层沟通,ViewModel 只可以通过一个或者多个 UseCase 去获取 repository,这种分层的做法有助于降低耦合度和提高可测试性,同时方便管理,让所有的 UseCase 都在耗时线程中执行,保证在 UI 线程中没有耗时数据的入口。

Google I/O 2018 app数据架构

展示层具体使用以下技术:Views + ViewModels + LiveData

ViewModels 从 LiveData 那里获取数据提供给 Views,通过 Data Binding 真正的 UI 控件就可以自动完成数据显示,可以从 Activity 和 Fragment 那些繁杂而单调的样式代码中解放出来。

我们以购买会员为例,说明如何使用 ViewModel、UseCase、Repository。首先涉及到 BuyVIPFragment.kt,VIPViewModel.kt,VIPViewModelFactory.kt,VIPRepository.kt 等类。

//购买 VIP Fragment 类
class BuyVIPFragment : Fragment() {

    private lateinit var vipViewModel: VIPViewModel

    override fun onActivityCreated(savedInstanceState: Bundle?) {
        super.onActivityCreated(savedInstanceState)
        activity?.let { activity ->
            //通过 ViewModelProviders.of() 获得 VIPViewModel 
            //因为使用的是 ViewModelProviders.of(activity) ,所以获取到的是上一个 Fragment 共有的 ViewModel,从而到达两个 Fragment 的数据可以进行同步更新 
            vipViewModel =
                ViewModelProviders.of(activity, VIPViewModelFactory(activity.application!!))
                    .get(VIPViewModel::class.java)

            //liveData 的监听回调方法(LiveData 用法不在这里详细讲解)
            vipViewModel.vipAction.observe(this, Observer<VIPDto> {
                btnBuyVIP.visibility = View.VISIBLE
                txtUserName.text = it.userName
                txtDeadlineDate.text = SimpleDateFormat("yyyy-MM-dd").format(it.deadlineDate)
            })
        }
        //触发购买 VIP 方法
        btnBuyVIP.setOnClickListener {
            btnBuyVIP.visibility = View.GONE
            vipViewModel.buyVIP()
        }
    }
}
//viewmodel 工厂生成类
class VIPViewModelFactory(val application: Application) : ViewModelProvider.Factory {
    override fun <T : ViewModel?> create(modelClass: Class<T>): T {
        //新建 VIPViewModel,并触发获取数据方法 getData()
        //网上有些例子是不使用 Factory,在获取 VIPViewModel 处,再调用初始化方法。这样也可以实现,不过如果是多个 Fragment 共用 ViewModel 就有可能会有数据重复加载的问题
        return VIPViewModel(application).also { it.getData() } as T
    }
}
//viewmodel 类
class VIPViewModel(application: Application) : AndroidViewModel(application) {

    private val _VIPAction = MediatorLiveData<VIPDto>()
    //对外 LiveData 类,ViewModel 与 LiveData 结合使用,方便 UI 控制器进行数据监听及更新,数据有更改时会通知观察者
    val vipAction: LiveData<VIPDto>
        get() = _VIPAction


    //获取数据
    fun getData() {
        //使用 UseCase 获取数据,一般 UseCase 的生命周期与 ViewModel 一样长,所以他们之间也可以不使用 LiveData , 不过这里也使用了 LiveData 进行转化了
        val useCase = GetVIPUseCase(SharePreferencesHelper(getApplication(), "VIP"))
        _VIPAction.addSource(useCase.observe()) {
            //一般会这里处理一些数据成功失败状态
            if (it.isSuccess) {
                _VIPAction.value = it.getOrThrow()
            }
        }
        useCase.execute(Unit)
    }

    //购买 VIP
    fun buyVIP() {
        val useCase = BuyVIPUseCase(SharePreferencesHelper(getApplication(), "VIP"))
        _VIPAction.addSource(useCase.observe()) {
            if (it.isSuccess) {
                _VIPAction.value = it.getOrThrow()
            }
        }
        _VIPAction.value?.let { useCase.execute(it) }

    }
}
//获得 VIP 用户数据
class GetVIPUseCase(private val sharePreferencesHelper: SharePreferencesHelper) : MediatorUseCase<Unit, VIPDto>() {
    override fun execute(parameters: Unit) {
        //从 Repository 获取数据
        val vipRepositoryData = VIPRepository(sharePreferencesHelper).getVIP()

        //一般还会有成功、失败两种状态
        val switchMapFunction = Function<VIPDto, LiveData<Result<VIPDto>>> {
            MediatorLiveData<Result<VIPDto>>().apply { value = Result.success(it) }
        }

        //敲黑板!!这是重点也是难点!!
        //1.为什么 UseCase 从 Repository 处获取数据要使用 LiveData?
        //因为生命周期的原因, UseCase 的生命周期一般与 ViewModel 一样长,
        //ViewModel 的生命周期一般会伴随 UI 控制器的销毁而销毁。
        //有时候获取数据时间会很漫长,漫长到 UI 控制器销毁了之后才返回来。
        //这时候就有可能会导致内存泄漏或者回调的时候会出现 View/ViewModel 不存在的情况。

        //2.在 UseCase 中如何使用 LiveData?
        //通过使用 Transformations.switchMap() 的目的是可以将上一层( UI 控制器对ViewModel 的 LiveData 观察状态传递给 UseCase 。
        //因为 LiveData.observe() 方法需要 LifecycleOwner,
        //但是 UseCase 没有 LifecycleOwner,
        //所以通过这种办法可以进行转化传递
        result.addSource(
            Transformations.switchMap(
                vipRepositoryData,
                switchMapFunction
            ) as MediatorLiveData<Result<VIPDto>>
        ) {
            result.value = it
        }
    }
}
//常见的 Repository 类
class VIPRepository(private val sharePreferencesHelper: SharePreferencesHelper) {

    fun buyVIP(user: VIPDto): LiveData<VIPDto> {
        val result = MutableLiveData<VIPDto>()

        //异步赋值的
        GlobalScope.launch(Dispatchers.Main) {
            async(Dispatchers.IO) {
                delay(1000)
                val calendar = Calendar.getInstance()
                calendar.time = user.deadlineDate
                calendar.add(Calendar.MONTH, 1)

                val deadLineDate = calendar.time.toString()
                val userName = user.userName

                val contentValue = SharePreferencesHelper.ContentValue("deadLineDate", deadLineDate)
                val contentValue2 = SharePreferencesHelper.ContentValue("userName", userName)
                sharePreferencesHelper.putValues(contentValue, contentValue2)
                async(Dispatchers.Main) {
                    result.value = VIPDto(userName, Date(deadLineDate))
                }
            }.await()
        }

        return result
    }

    fun getVIP(): LiveData<VIPDto> {

        val result = MutableLiveData<VIPDto>()
        //异步赋值的
        GlobalScope.launch(Dispatchers.Main) {
            async(Dispatchers.IO) {
                delay(1500)
                val deadLineDate = sharePreferencesHelper.getString("deadLineDate") ?: Date().toString()
                val userName = sharePreferencesHelper.getString("userName") ?: "EdgarNg"
                async(Dispatchers.Main) {
                    result.value = VIPDto(userName, Date(deadLineDate))
                }
            }.await()
        }

        return result
    }
}

通过查看代码可以知道,最后的状态是下图这样的:

kjaeSI.png

这里留个小功课,当 APP 在购买 VIP 页面,翻转屏幕后,APP 会”回到“上一个页面,请回答为什么,并提出解决办法。

ViewModel 的 Unit test (TODO)

内部实现

偏向源码以及设计思想分析,不感兴趣的同学可以跳过这章,谢谢!

不过真得很精彩的窝,真得不看看吗?

源码来源编文时最新 lifecycle-viewmodel-2.1.0-alpha02-sources.jar,截图已经附上行号,方便比对

为了更好地详细了解 ViewModel,我们可以带着几个问题去学习:

  1. ViewModel 是怎么生成的?
  2. ViewModel 为什么可以达到这么神奇的效果,Activity/Fragment 都重新生成居然还可以活下来,它是怎么做到的?

1. 常见的 ViewModel 实现方式:

val ViewModel1 = ViewModelProviders.of(activity,factory).get(ViewModel::class.java)
val ViewModel2 = ViewModelProviders.of(fragment,factory).get(ViewModel::class.java)

ViewModel 主要实现机制采用了工厂模式,传入 Factory 去实例化指定的ViewModel类,如果不传 Factory,则使用默认的,如下图。


默认的 Factory

具体的调用流程如下,由此可以看出ViewModel的生成其实还是由我们传入的 factory.create() 方法来生成

ViewModel 生成过程

Ps:请注意 ViewModelProvider 和 ViewModelProviders,是两个独立的类。具体可以从下列截图看出,ViewModelProviders 是 LifeCycle 库的扩展类,高内聚低耦合,核心的东西放在一个类,有可能变化的扩展的东西放在另外一个地方进行扩展,这也可以启发到我们新手在设计框架时要考虑到扩展性,尤其是做 SDK 或者底层架构的同学可以参考学习这一块的设计方式。

ViewModelProvider 和 ViewModelProviders 的区别.png

2.为什么 ViewModel 的生命周期这么长(比一般的 UI 数据长)

ViewModel 生命周期

首先我们重温一下 ViewModel 的生命周期(如上图),ViewModel 的生命周期不会因为配置变更(Activity 的多次调用onCreate()方法)会终止,反而会和 Activity 的 onDestroy()方法一起结束。为什么可以这样子?!请接着往下看。

kcd0C6.png

要想知道 ViewModel 的生命周期,思路上当然是从 ViewModel 那里获得答案(其实从 UI 控制器那里获取更快,不过这算作弊)。根据 ViewModel 生成的流程图可以知道,是调用了 ViewModelProvider.get(@NonNull String key, @NonNull Class<T> modelClass) 返回生成 ViewModel 的,查看源码 get 方法(上图)可以知道,一共有两处地方生成 ViewModel 的,ViewModel viewModel = mViewModelStore.get(key);viewModel = mFactory.create(key, modelClass); ,但是生成完了之后都通过 mViewModelStore.put(key, viewModel); ,通过方法名称应该就是一个缓存器,查看 mViewModelStore.put(key, viewModel)的源码后,发现就是常见的内部缓存了一个Map进行对ViewModel的管理。

ViewModelStore

那么这个 mViewModelStore 是什么为什么要缓存 viewModel

这个 mViewModelStore 就是实例化 ViewModelProvider 时候传进来的(请见 ViewModel 生成的流程图,由 activity.getViewModelStore()fragment.getViewModelStore() 获得)。

kcwoJx.png

了解 mViewModelStore 是什么之后,我们就解决第二个疑问? UI 控制器中的属性 mViewModelStore 为什么要缓存viewModel?难道是想重新拿出来用,来达到 ViewModel 的长生命周期?(因为刚刚上上图提到第一处生成 ViewModel 的就是从 mViewModelStore 那里 get(key) 出来的)接下来我们就去了解 mViewModelStore 在 Activity 和 Fragment 中的作用了,我们先了解 Activity 的。

在 Activity 中能够看到 getViewModelStore() 方法,mViewModelStore 有三种方式,1.自身不为 null ;2. nc.viewModelStore ;3. new ViewModelStore();。通过查看 mViewModelStore 在整个 Activity 类中,只有这个方法用到,所以 mViewModelStore 只能从2 3中获得实例。3不用说,直接看2 --> getLastNonConfigurationInstance() 从名字上看相当有嫌疑!

kc0zB4.png

点击 getLastNonConfigurationInstance() 查看源码,发现描述如下图。通过描述可以知道,我们没有找错!!获得之前由{@link #onRetainNonConfigurationInstance()}返回的 non-configuration instance data 。这将从初始化 {@link #onCreate} 和 {@link #onStart} 调用新实例中获得 non-configuration instance data ,允许您从前一个实例中提取任何有用的动态状态。话虽如此,但是还是想知道 mLastNonConfigurationInstances 它是怎么来的?为什么可以把上一个 Activity 的 NonConfigurationInstances 数据拿到现在的 Activity 来

kcDusU.png

根据描述,是从 onRetainNonConfigurationInstance() 方法那里来的,但是查看了 Activity 的这个方法,发现方法里面是 null !然后难道是子类做了实现?为什么会这么想?

kgIaA1.png
kc0zB4.png
kg7N7j.png

因为 public static ViewModelProvider of(@NonNull FragmentActivity activity,@Nullable Factory factory) 中使用的是 FragmentActivity 而不是 Activity ,在方法设计上会偏向使用基类来进行参数传递。而且之前的 mViewModelStore 也是由 FragmentActivity 的父类 ComponentActivity 生成的,而且 NonConfigurationInstances 类也是 ComponentActivity 的内部类。所以我们这才推断是不是 Activity 的子类实现了。同时我们也可以学习到:

  1. 要对 ActivityComponentActivityFragmentActivityAppCompatActivity 之间的异同有所涉猎,尤其在设计底层框架以及接口的时候,要慎重选择,不然会设计出来不好用,处处受阻。
  2. 设计底层的时候,类适当的分层继承有助于架构的清晰解耦。

我们只需查看 FragmentActivityComponentActivity 哪个类实现了 onRetainNonConfigurationInstance() 就可以了。

查看 FragmentActivity 的父类 ComponentActivity 的时候可以见到

TODO:
已知是调用ComponentActivity的onRetainNonConfigurationInstance()来保存了NonConfigurationInstances数据的,这就涉及到APP的启动、AMS等一系列比较复杂的过程,推荐点击ConfigurationChanged流程梳理(屏幕旋转、语言及字体切换)

  1. 这个方法是什么时候被调用的?
  2. 它是如何将数据给下一个Activity的?

所以这也同时回答了“ ViewModel 为什么不能绑定 Activity、Fragment 的问题?

而 ViewModel 什么时候才会被 clear 掉呢?根据 ViewModel 的生命周期图标,应该是 activity 调用 onFinish() 会调用 ViewModel 的 clear()。我们进一步查看哪里调用了 ViewModel 的 clear() 方法,可知会从 ComponentActivity.JavaFragmentManagerViewModel.java 处有所调用,我们先分析 ComponentActivity.Java

由下图可以看到,当 ComponentActivity 是 ON_DESTROY 状态时,而且不是配置变更时,会调用 ViewModelStore.clear() 方法清空与之绑定的 ViewModel 数据。

ComponentActivity清空数据

由下图 FragmentManagerViewModel 清空数据分析可见,当 Fragment 应该 Destroy 时,而且不是配置变更时,会调用 ViewModelStore.clear() 方法清空与之绑定的 ViewModel 数据。

k8zBnO.png
k8z29I.png

所以这就解析了,ViewModel 的生命周期为什么可以跟 Activity 一样长,以及为什么可以在配置变更后还可以继续复用了。

总结

  1. ViewModel 本身只是一个保存数据的类,需要搭配 livedata、databinding 才能发挥效果。单独用作用不大,不过好处是解耦,方便 unit test,效果很明显!这篇只是推广初级文章,很多注意的细节点,可以点击这里
  2. ViewModel 尽量不要绑定跟Android包有关的类,有可能造成内存泄漏(具体请见[内部实现]章节)以及会给 Unit Test 造成困难,Google 建立 JetPack 的其中一个目的就是解耦可测试!

源码地址

https://github.com/EdgarNg1024/kaixue-docs/tree/master/viewmodel

https://github.com/google/iosched/

参考

  1. Android Jetpack - ViewModel | 中文教学视频
  2. ViewModels and LiveData: Patterns + AntiPatterns (译文)[强烈推荐,作者应该是 Google 里面有参与 ViewModel 的实现开发的,所以文章的推荐用法应该是最贴近 ViewModel 开发思想]
  3. ViewModels: Persistence, onSaveInstanceState(), Restoring UI State and Loaders (译文)[强烈推荐,作者应该是 Google 里面有参与 ViewModel 的实现开发的,所以文章的推荐用法应该是最贴近 ViewModel 开发思想]
  4. Google I/O 2018 app — Architecture and Testing[强烈推荐,作者应该是 Google 里面有参与 ViewModel 的实现开发的,所以文章的推荐用法应该是最贴近 ViewModel 开发思想]
  5. SharePreference数据存储工具类[借用了这哥们的SharePreference工具类]

撰稿人

Edgar Ng(https://github.com/EdgarNg1024)

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