Android的ViewModel

这篇具有很好参考价值的文章主要介绍了Android的ViewModel。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

前言

在Compose的学习中,我们在可组合函数中使用rememberSaveable​​​​​​​保存应用数据,但这可能意味着将逻辑保留在可组合函数中或附近。随着应用体量不断变大,您应将数据和逻辑从可组合函数中移出。

而在之前的应用架构学习中,我们接触到了MVVM架构,他将应用数据存储在了ViewModel中,它是Android Jetpack 库中的架构组件之一。

当框架在配置更改或其他事件期间销毁并重新创建 activity 时,存储的数据不会丢失。不过,如果 activity 因进程终止而被销毁,数据将会丢失。ViewModel 只能通过快速重新创建 activity 缓存数据。

了解应用架构

架构原则

最常用的架构原则包括:分离关注点通过模型驱动界面

  • 分离关注点:该原则指出,应将应用分为函数类,每个类有各自的职责
  • 通过模型驱动界面:该原则指出,应该通过模型驱动界面,最好是持久性模型

模型是负责处理应用数据的组件。它们独立于应用中的界面元素和应用组件,因此不受应用的生命周期以及相关的关注点的影响。

推荐的应用架构 

基于上面两点原则,每个应用应至少有两个层:

  • 界面层:屏幕上显示应用数据,但独立于数据层的层
  • 数据层:用于存储、检索和提供应用数据的层

每当数据因用户互动(例如按了某个按钮)而发生变化时,界面都应随之更新,以反映这些变化。

界面层由以下组件组成:

  • 界面元素:用于在屏幕上呈现数据的组件。您将使用 Compose 构建这些元素。
  • 状态容器:用于保存数据、向界面提供数据以及处理应用逻辑的组件。此处我们使用 ViewModel

Android的ViewModel,android

ViewModel

简介

ViewModel组件用于存储和公开界面所使用的状态(UI State)。

界面状态(UI State)是经过ViewModel转换的应用数据。界面(UI)是相对于用户而言的,界面状态是相对于应用而言的,例如一个开关switch展现在用户面前,而switch是开还是关,就是switch的界面状态。因此,对于界面状态的任何改变,都会直接影响界面。

ViewModel会存储应用相关的数据,这些数据不会在activity被销毁并重新创建时被销毁。应用会在配置更改期间自动保留ViewModel对象,以便在重组时ViewModel存储的数据可以立即被使用。

与Compose梦幻联动

在使用Compose时,ViewModel是向可组合项展示界面状态的主要方式。过去我们将数据的存储和处理方式留在activity或fragment中,臃肿而不直观;如今在混合应用中,activity和fragment仅用于托管可组合函数。

添加ViewModel

以下是用户掷骰子屏幕的 ViewModel 实现示例。

1. 打开app模块下的build.gradle.kts,在dependencies块添加如下内容:

dependencies {
// other dependencies

    implementation("androidx.lifecycle:lifecycle-viewmodel-compose:2.6.1")
//...
}

2. 添加一个数据类存储各项值,并创建一个类继承ViewModel:

//数据类,存储游戏的值
data class DiceUiState(
    val firstDieValue: Int? = null,
    val secondDieValue: Int? = null,
    val numberOfRolls: Int = 0,
)

class DiceRollViewModel : ViewModel() {

    // 展示界面状态
    // 此处的StateFlow是数据容器式可观察数据流,其value属性反映了当前的状态值
    // 有了它,可组合函数就可以监听界面状态更新

    //防止外部类修改ViewModel的数据,设置为private,同时val类型不包含setter,为只读属性
    private val _uiState = MutableStateFlow(DiceUiState())
    //asStateFlow方法使可变状态流变为只读状态流,界面通过只读属性的uiState读取值
    val uiState: StateFlow<DiceUiState> = _uiState.asStateFlow()

    // 处理业务逻辑
    fun rollDice() {
        _uiState.update { currentState ->
            //调用 copy 方法来创建新的 DiceState 对象,并将其赋值给 uiState 变量。
            //这将触发 UI 的重新渲染。
            currentState.copy(
                firstDieValue = Random.nextInt(from = 1, until = 7),
                secondDieValue = Random.nextInt(from = 1, until = 7),
                numberOfRolls = currentState.numberOfRolls + 1,
            )
        }
    }
}

3. 从activity访问ViewModel:文章来源地址https://www.toymoban.com/news/detail-834088.html

import androidx.activity.viewModels

class DiceRollActivity : AppCompatActivity() {

    override fun onCreate(savedInstanceState: Bundle?) {
        // 在系统第一次调用Activity的onCreate()方法时创建一个 ViewModel实例
        // 重新创建的activity会收到相同的、第一次创建activity时留下的ViewModel实例

        // 此处使用了'by viewModels()'的Kotlin属性委托
        // 他创建并初始化与activity相关联的ViewModel
        val viewModel: DiceRollViewModel by viewModels()
        lifecycleScope.launch {
            repeatOnLifecycle(Lifecycle.State.STARTED) {
                viewModel.uiState.collect {
                    // 在此处更新ui元素
                }
            }
        }
    }
}

到了这里,关于Android的ViewModel的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包赞助服务器费用

相关文章

  • Android Jetpack 从使用到源码深耕【ViewModel从实践到原理 】(二)

    上文,我们对ViewModel的引入背景、使用、进行了实例编码、总结分析,本文开始,我们对ViewModel的实现源码,进行探索、总结。 Android Jetpack 从使用到源码深耕【ViewModel从实践到原理 】(一) Android Jetpack 从使用到源码深耕【ViewModel从实践到原理 】(二) Android Jetpack 从使用

    2024年02月04日
    浏览(10)
  • Android Jetpack 从使用到源码深耕【ViewModel从实践到原理 】(三)

    上文,我们通过简单的ViewModel使用源码入手,对其源码进行阅读,原理进行了简单总结,简单来说,ViewModel是通过Activity的onRetainNonConfigurationInstance 与 getLastNonConfigurationInstance的自动调用,实现了 ViewModel数据的存储和恢复,数据存储在ViewModelStore的map中,更加明确的话,应该是

    2024年02月04日
    浏览(36)
  • Android:单Activity多Fragment,Navigation实现Fragment跳转,Fragment之间通过ViewModel共享数据

    Android:单Activity多Fragment,Navigation实现Fragment跳转,Fragment之间通过ViewModel共享数据

    1、activity_main.xml 2、MainActivity FragmentA:包括SeekBar和一个按钮,点击button跳转到FragmentB FragmentB:SeekBar加一和减一操作的按钮,一个返回FragmentA的按钮,即经过加减操作以后,在FragmentA上显示加减的结果。 nav_graph.xml 1、fragment_home.xml 2、HomeFragment 1、fragment_detail.xml 2、DetailFragm

    2023年04月08日
    浏览(14)
  • 【Jetpack】ViewModel 架构组件 ( 视图 View 和 数据模型 Model | ViewModel 作用 | ViewModel 生命周期 | 代码示例 | 使用注意事项 )

    【Jetpack】ViewModel 架构组件 ( 视图 View 和 数据模型 Model | ViewModel 作用 | ViewModel 生命周期 | 代码示例 | 使用注意事项 )

    Activity 遇到的问题 : 瞬态数据丢失 : 操作 Activity 时 , 如果 屏幕 自动旋转 , 当前 Activity 组件会 执行销毁操作 , 并重新创建新的 Activity 组件 , 该操作会 导致 Activity 的 瞬态数据 丢失 ; 内存泄漏 : 在 系统组件 如 Activity 中 , 启动了一个线程 , 在线程中执行一系列操作 , 如果 A

    2024年01月25日
    浏览(9)
  • LiveData和ViewModel源码学习

    LiveData ObserverWrapper 数据观察者 LifecycleEventObserver 生命周期观察者 活跃状态接受并更新数据 刷新页面数据 非活跃状态暂停接收数据 防止崩溃, 销毁状态的时候移除观察者 防止内存泄漏 数据可以监听 数据倒灌,就是先发送数据,通过生命周期响应来触发change 数据粘性,先发

    2024年04月12日
    浏览(11)
  • 沉思篇-剖析Jetpack的ViewModel

    沉思篇-剖析Jetpack的ViewModel

    ViewModel做为架构组件的三元老之一,是实现MVVM的有力武器。 ViewModel的基本功能就是管理UI的数据。其实,从职责上来说,这又是对Activity和Fragment的一次功能拆分。以前存储在它们内部的数据,需要它们自己处理创建,更新,存储,恢复的所有过程,同时它们还要处理UI的数据

    2024年02月08日
    浏览(13)
  • ViewModel和LiveData的基本使用

    ViewModel和LiveData的基本使用

    在页面(活动/碎片)中,在很简单的情况在我们会将UI交互,数据的获得与处理等相关的逻辑,全都放在一个页面中,但是如果我们要处理的业务很复杂,这样做会显得代码冗杂且不利于解读,这样不符合\\\" 单一责任 \\\"原则. 所以页面就应该只是负责接收用户的交互以及数据的展示,其他逻

    2024年02月06日
    浏览(9)
  • 王学岗码牛jetpack系列之ViewModel

    王学岗码牛jetpack系列之ViewModel

    ViewModel的目的:存储数据,以注重生命周期的方式管理界面的相关数据 viewModel的特性:1,数据持久化,不依赖于Activity的生命周期,有自己独立的生命周期 2,异步回调不会造成内存泄漏 3,隔离Model层与View层 4,Fragments间共享数据 我们看一段代码 看下打印结果 Activity不一样了,

    2024年02月02日
    浏览(15)
  • SavedStateHandle的介绍----ViewModel不具备保存状态数据的功能

    SavedStateHandle的介绍----ViewModel不具备保存状态数据的功能

    LiveData本身不能在进程销毁中存活,当内存不足时,Activity被系统杀死,ViewModel本身也会被销毁。 为了保存LiveData的数据,使用SavedStateHandle。 事故场景: 进程销毁后,重新去通过ViewModel去获取LiveData数据,获取到的数据是null。 这表示ViewModel不具备onSavedInstance的功能。   引入

    2024年02月16日
    浏览(14)
  • Avalonia中如何将View事件映射到ViewModel层

    前面的文章里面我们有介绍在Wpf中如何在View层将事件映射到ViewModel层的文章,传送门,既然WPF和Avalonia是两套不同的前端框架,那么WPF里面实现模式肯定在这边就用不了,本篇我们将分享一下如何在Avalonia前端框架下面将事件映射到ViewModel层。本章内容还是在上一节的基础上

    2024年02月04日
    浏览(13)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包