初探Jetpack(一) -- ViewModel

一、JetPack 简介

以前在没有 Jetpack 的时候,大家项目中最常使用的就是 MVP 或者 MVVM 等设计模式来解耦逻辑和UI,当然这非常考验个人的设计能力和经验。
Google 貌似也意识到了这个问题,在 2018推出了一个全新的架构组件库 – Jetpack。
Jetpack 适合一个开发组件工具库,它的主要目的是帮助我们编写出更加简洁的代码,并简化我们的开发开发特点;Jetpack 它不依赖任何 Android 系统版本,它定义在 androidx 中,并拥有非常好的向下兼容。

首先,看一张 Jetpack 的全家桶:
在这里插入图片描述
官网文档:Jetpack

可以看到 Jetpack 几乎覆盖了开发所有的部分;目前 Google 主推的架构就是 MVVM,所以 Jetpack 其实就是专门为 MVVM 设计的架构组件。

都 2020 了,那肯定也要学习一下 Jetpack 组件了,开冲!!

首先,先学习 ViewModel,

二、ViewModel 介绍

在传统开发模式下,Activity、fragment 等UI任务繁琐,又要负责逻辑,也要负责 UI,往往在一个比较大型的项目,一个 Activity 几千行代码算少的,让人看了就没下手的欲望,这也为啥衍生了 MVP、MVVM 的原因;
而使用 ViewModel 的一个很重要的作用,就是可以为 Activity 分担一部分工作,它是专门用于存放于界面相关的数据的。也就是说,只要是界面上能看到的东西,它的变量都应该放在 ViewModel 中,而不是 Activity,这样就可以一定程序上减少 Activity 的逻辑。

另外,ViewModel 还有一个特性,就是当手机旋转等非正常退出时,它自身不会重建,即数据不会丢失,比如 Activity 在屏幕旋转时,会重建,从而导致数据丢失,虽然可以使用 onSaveInstanceState() 从onCreate()方法获取,但通过 ViewModel 就不用担心这个。
ViewModel 的生命周期如下:
在这里插入图片描述

三、基本用法

首先你需要创建一个 androidx 的工程,如果要使用 ViewModel,还需要 关联以下依赖:

    implementation 'androidx.lifecycle:lifecycle-extensions:2.2.0'

通常来讲,比较好的规范就是给 每一个 Activity 或者 Fragment 创建一个对应的 ViewModel。
所以,这里我们也创建一个于 activity 对应的 MainViewModel,并让它集成 ViewModel,如:

class MainViewModel() : ViewModel() {
    var counter = 0;
}

xml 中写一个按钮,每次就加1,并通过 textview 显示出来

    <TextView
        android:id="@+id/infoText"
        android:layout_width="match_parent"
        android:layout_height="30dp"
        android:gravity="center"
        />
    <Button
        android:id="@+id/textcounter"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:text="Plus one"
        />

接着,则是通过ViewModelProvider 创建 ViewModel 的实例,由于 ViewModel 有自己的生命周期,并且生命周期长于activity,如果再 oncreate 创建实例,这样每次都会创建一个新的实例,这样就无法保存数据了。所以,ViewModel 的实例创建应该这样:

 viewModel = ViewModelProvider(this).get(MainViewModel::class.java)

所以,完整程序如下:

 val viewModel = ViewModelProvider(this).get(MainViewModel::class.java)

 textcounter.setOnClickListener{
     viewModel.counter ++;

     infoText.text = viewModel.counter.toString()
 }

 infoText.text = viewModel.counter.toString()

效果如下:
![在这里插入图片描述](https://img-blog.csdnimg.cn/2020061107521018.png在这里插入图片描述
可以看到,当旋转屏幕,数据也不会发现变化

3.1 向 ViewModel 传递参数

上一节,ViewModel 并没有构造参数,如果我们要想它传递初始值怎么办呢?比如传递一个数值。

这个问题,可以通过 ViewModelProvider.Factory 来实现,首先,修改MainViewModel,如:

class MainViewModel(count:Int ):ViewModel() {
    var counter = count
}

然后新建一个 MainViewModelFactory 并让它实现 ViewModelProvider.Factory 接口,如下:

class MainViewModelFactory(val count:Int):ViewModelProvider.Factory {
    override fun <T : ViewModel?> create(modelClass: Class<T>): T {

        return MainViewModel(count) as T
    }
}

可以看到,MainViewModelFactory 的构造参数接受一个 count,并在 create 的时候,对 MainViewModel 初始化并传递数值。

接着,修改一下代码:

ViewModelProvider(this,MainViewModelFactory(5)).get(MainViewModel::class.java)

可以看到数值一开始就改变了:
在这里插入图片描述

后面的文章,我们再继续深入了解 ViewModel

参考:<第一行代码> 第三版

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章