安卓MVI架构真的来了?动手试着封装吧(二)上

简介: 安卓MVI架构真的来了?动手试着封装吧(二)

前言:由于框架本身也在不断地迭代,因此文章中的部分代码可能存在更新或者过时,如果你想阅读源码或者查看代码的在项目中的实际使用方法,可以查看笔者目前在维护的compose项目:Spacecraft: 《Spacecraft - 我的安卓技术实践平台》-查看代码请进入develop分支 (gitee.com)

本篇内容主要为如何把ViewModel改造成状态和事件的容器


当状态遇上了事件


  • 状态
    View在监听UI状态的时候,会记录初始值并初始化自身,如果后续的监听中,View发现了UI状态的值与之前发生了变化,就会更新自身(或者调用与之相关的逻辑代码)

image.png

image.png

  • 当状态中混杂了事件

image.png  可见,如果你把事件(例如toast事件)也当成UI状态的一部分的时候,那么这个只需要显示一次的toast就会被携带到下次UI重构中,以一种“状态”的方式复活,即数据倒灌。


事件与状态分手吧


  如果我们不希望事件以一种状态的方式被倒灌到下次UI重构中,就把它从状态中提取出来吧。在谷歌的开发者文档中,UiState是用kotlin语言中的StateFlow实现的,而StateFlow是一种特殊的SharedFlow,因此使用SharedFlow来表示事件流,而状态就用原本的StateFlow。

如果你对SharedFlow和StateFlow还不是特别理解,可以参考以下的文档(为掘金其他优秀作者撰写,仅供参考):


@Keep
interface UiState
@Keep
interface UiSingleEvent

  我并不希望对ViewModel进行直接封装成容器本身,即实现一套类似BaseViewModel的东东,这种继承重写的方式也许可以实现我的需求,但是过于生硬,也不容易对现有的代码进行重构(修改基类过于蛋疼),因此我更希望容器和viewModel的关系更像activity和viewModel的关系。


//activity中的viewModel
private val viewModel by viewModels<FriendViewModel>()
//希望实现类似的api
private val container by containers()

  接下来,写一个容器接口,容器拥有状态和事件的流。


/**
 * 状态容器,分别存储UI状态和单次事件,如果不包含单次事件,则使用[Nothing]
 */
interface Container<STATE : UiState, SINGLE_EVENT : UiSingleEvent> {
    //ui状态流
    val uiStateFlow: StateFlow<STATE>
    //单次事件流
    val singleEventFlow: Flow<SINGLE_EVENT>
}

  我们对原来的接口增加一个直接子类,增加2个修改方法。这样做参考了List和MutableList的关系,MutableContainer是对内提供的(例如给viewModel使用),允许取值和修改;Container是对外提供的(例如activity,fragment等),只允许取值,这样避免了UI绕过viewModel直接修改UI状态的值,确保数据单向流动。


interface MutableContainer<STATE : UiState, SINGLE_EVENT : UiSingleEvent> :
    Container<STATE, SINGLE_EVENT> {
    //更新状态
    fun updateState(action: STATE.() -> STATE)
    //发送事件
    fun sendEvent(event: SINGLE_EVENT)
}

  最后,得出一个实现类,RealContainer,其中UI状态使用了StateFlow,UI事件则使用SharedFlow(确保事件不会倒灌)。


internal class RealContainer<STATE : UiState, SINGLE_EVENT : UiSingleEvent>(
    initialState: STATE,
    private val parentScope: CoroutineScope,
) : MutableContainer<STATE, SINGLE_EVENT> {
    private val _internalStateFlow = MutableStateFlow(initialState)
    private val _internalSingleEventSharedFlow = MutableSharedFlow<SINGLE_EVENT>()
    override val uiSteFlow: StateFlow<STATE> = _internalStateFlow
    override val singleEventFlow: Flow<SINGLE_EVENT> = _internalSingleEventSharedFlow
    override fun updateState(action: STATE.() -> STATE) {
        _internalStateFlow.update { action(_internalStateFlow.value) }
    }
    override fun sendEvent(event: SINGLE_EVENT) {
        parentScope.launch {
            _internalSingleEventSharedFlow.emit(event)
        }
    }
}


到这里,其实容器已经可以直接使用的了:在ViewModel中新增一个成员变量,然后new一个容器对象,viewModel对容器对象进行赋值操作,View层分别订阅2个流即可完成状态和事件的分离,即可完成状态和事件的分离。


相关文章
|
7月前
|
前端开发 数据可视化 Java
Android用Canvas画一个折线图,并加以简单封装
本文介绍了如何用Java绘制动态折线图,从固定折线图的实现到封装成可复用的组件。首先通过绘制XY坐标轴、添加坐标标签和绘制折线及数据点完成基础折线图。接着,将静态数据替换为动态输入,支持自定义X轴、Y轴和折线数据。代码中包含关键方法如`drawDaxes`(绘制坐标轴)、`drawAxispoint`(绘制坐标点)和`drawbrokenLine`(绘制折线)。最终实现可根据传入数据动态生成折线图,适用于Android开发中的数据可视化场景。
257 0
|
安全 Android开发 iOS开发
深入探索Android与iOS的差异:从系统架构到用户体验
在当今的智能手机市场中,Android和iOS无疑是最受欢迎的两大操作系统。本文旨在探讨这两个平台之间的主要差异,包括它们的系统架构、开发环境、安全性、以及用户体验等方面。通过对比分析,我们可以更好地理解为何不同的用户群体可能会偏好其中一个平台,以及这些偏好背后的技术原因。
|
Android开发 Swift iOS开发
深入探索iOS与Android操作系统的架构差异及其对应用开发的影响
在当今数字化时代,移动设备已经成为我们日常生活和工作不可或缺的一部分。其中,iOS和Android作为全球最流行的两大移动操作系统,各自拥有独特的系统架构和设计理念。本文将深入探讨iOS与Android的系统架构差异,并分析这些差异如何影响应用开发者的开发策略和用户体验设计。通过对两者的比较,我们可以更好地理解它们各自的优势和局限性,从而为开发者提供有价值的见解,帮助他们在这两个平台上开发出更高效、更符合用户需求的应用。
|
10月前
|
Android开发 开发者 Kotlin
Android实战经验之Kotlin中快速实现MVI架构
MVI架构通过单向数据流和不可变状态,提供了一种清晰、可预测的状态管理方式。在Kotlin中实现MVI架构,不仅提高了代码的可维护性和可测试性,还能更好地应对复杂的UI交互和状态管理。通过本文的介绍,希望开发者能够掌握MVI架构的核心思想,并在实际项目中灵活应用。
468 8
|
网络协议 Linux Android开发
深入探索Android系统架构与性能优化
本文旨在为读者提供一个全面的视角,以理解Android系统的架构及其关键组件。我们将探讨Android的发展历程、核心特性以及如何通过有效的策略来提升应用的性能和用户体验。本文不包含常规的技术细节,而是聚焦于系统架构层面的深入分析,以及针对开发者的实际优化建议。
349 21
|
存储 Linux API
深入探索Android系统架构:从内核到应用层的全面解析
本文旨在为读者提供一份详尽的Android系统架构分析,从底层的Linux内核到顶层的应用程序框架。我们将探讨Android系统的模块化设计、各层之间的交互机制以及它们如何共同协作以支持丰富多样的应用生态。通过本篇文章,开发者和爱好者可以更深入理解Android平台的工作原理,从而优化开发流程和提升应用性能。
|
安全 Android开发 iOS开发
深入探索iOS与Android系统架构差异及其对开发者的影响
本文旨在通过对比分析iOS和Android两大移动操作系统的系统架构,探讨它们在设计理念、技术实现及开发者生态方面的差异。不同于常规摘要仅概述内容要点,本摘要将简要触及核心议题,为读者提供对两大平台架构特点的宏观理解,铺垫
|
IDE 安全 Android开发
深入探索Android与iOS操作系统的架构差异
本文旨在对比分析Android和iOS两大主流移动操作系统在架构设计上的根本差异。通过详细解读两者的系统架构、开发环境、以及安全性等方面,揭示它们各自的特点及优势,为开发者选择合适的平台提供参考。
|
Java Linux Android开发
深入探索Android系统架构:从Linux内核到应用层
本文将带领读者深入了解Android操作系统的复杂架构,从其基于Linux的内核到丰富多彩的应用层。我们将探讨Android的各个关键组件,包括硬件抽象层(HAL)、运行时环境、以及核心库等,揭示它们如何协同工作以支持广泛的设备和应用。通过本文,您将对Android系统的工作原理有一个全面的认识,理解其如何平衡开放性与安全性,以及如何在多样化的设备上提供一致的用户体验。
|
安全 Android开发 iOS开发
深入探讨Android与iOS的系统架构差异
本文旨在通过对比分析Android和iOS两大移动操作系统的系统架构,揭示它们在设计理念、安全性、应用生态及开发环境等方面的显著差异。我们将从底层架构出发,逐步剖析至用户界面层面,为开发者和科技爱好者提供一份详尽的技术参考。
717 1

热门文章

最新文章