MVVM 到 MVI:Android 架构迁移指南

关注个人简介,技术不迷路~

从模型-视图-视图模型 (MVVM) 架构迁移到模型-视图-意图 (MVI) 架构似乎是一项艰巨的任务。但凭借更清晰的架构和更可预测的应用程序行为的好处,这是一次非常值得付出努力的转变。

在本文中,我们将探讨 MVVM 和 MVI 之间的差异,并提供有关如何迁移到 MVI 的指导。

MVVM 与 MVI

MVVM 架构是移动应用程序开发中广泛使用的模式。它将用户界面(View)与数据和逻辑(ViewModel)分开,并允许两个组件之间进行数据绑定。ViewModel 负责为 View 准备数据并处理用户输入。

另一方面,MVI 架构是近年来流行的一种较新的模式。它还将视图和数据分开,但添加了一个称为意图的新层,该层负责以结构化且可预测的方式表示用户操作。Intent 被传递给模型,模型产生一个新的状态,然后显示在视图中。

两种架构之间的主要区别在于,在 MVVM 中,ViewModel 负责用户操作和更新视图状态。在 MVI 中,意图是一个代表用户操作的独立层,可以更轻松地推断应用程序行为并处理边缘情况。

为什么选择 MVI?

1.更好的关注点分离: 使用MVI,Intent层负责用户操作,Model层负责状态管理,View层负责状态渲染。这种分离使得推断应用程序行为以及对应用程序进行更改变得更加容易,而不会影响架构的其他部分。

2. 更可预测的应用程序行为: 使用 MVI,应用程序的状态始终由单个数据对象表示,这使得更容易推理和预测应用程序的行为。这种可预测性可以减少错误并提供更好的用户体验。

3. 提高可测试性: MVI 使编写单元测试变得更加容易,因为每一层都可以独立测试。通过架构的可预测数据流使得编写涵盖所有可能状态和边缘情况的测试变得更加容易。

如何从 MVVM 迁移到 MVI?

从 MVVM 迁移到 MVI 似乎是一项艰巨的任务,但重要的是要记住这是一个渐进的过程。以下是一些需要考虑的步骤:

  1. 了解MVI架构: 在进行切换之前,请确保您了解MVI的关键概念,包括Intent层和单向数据流。
  2. 识别组件: 仔细查看现有的 MVVM 代码并识别需要更改的组件。这包括 ViewModel、View 以及任何数据模型或存储库。
  3. 创建 Intent 层:将 Intent 层添加到您的应用程序中,该层以结构化且可预测的方式表示用户操作。该层应该与 ViewModel 和 Model 分开。
  4. 更新 ViewModel: 修改您的 ViewModel 以接受 Intent 并将其传递给模型。ViewModel 不应再负责更新 View 的状态
  5. 更新模型: 修改模型以根据意图和当前状态生成新状态。模型不应再负责处理用户输入。
  6. 更新视图: 修改视图以呈现模型提供的状态。视图不应再负责获取数据或更新状态。
  7. *****为架构的每一层* **编写单元测试,以确保数据流可预测,并覆盖所有可能的状态和边缘情况。****

从 MVVM 迁移到 MVI 的实施主要步骤

第 1 步:创建 Intent 类


sealed class MyIntent {  
object LoadData : MyIntent()  
data class UpdateData(val newData: Data) : MyIntent()  
object DeleteData : MyIntent()  
}

第2步:修改ViewModel

修改您的 ViewModel 以接受意图并将其传递给模型。ViewModel 不应再负责更新 View 的状态。

class  MyViewModel ( private  val myModel: MyModel) : ViewModel() { 
    private  val _state = MutableLiveData<MyState>() 
    val state: LiveData<MyState> = _state 

    fun  processIntent (intent: MyIntent ) { 
        val newState = myModel.processIntent(intent, _state.value) 
        _state.postValue(newState) 
    } 
}

第三步:创建State类

创建一个表示应用程序当前状态的类,例如:

class MyModel(private val myRepository: MyRepository) {
    fun processIntent(intent: MyIntent, currentState: MyState?): MyState {
        return when (intent) {
            is MyIntent.LoadData -> {
                // Load data from repository
                myRepository.getData()
                    .map { data -> currentState?.copy(data = data) ?: MyState(data = data) }
                    .onStart { emit(currentState?.copy(isLoading = true) ?: MyState(isLoading = true)) }
                    .onErrorReturn { throwable ->
                        currentState?.copy(isLoading = false, error = throwable)
                            ?: MyState(isLoading = false, error = throwable)
                    }
            }
            is MyIntent.UpdateData -> {
                // Update data in repository
                myRepository.updateData(intent.newData)
                currentState?.copy(data = intent.newData) ?: MyState(data = intent.newData)
            }
            is MyIntent.DeleteData -> {
                // Delete data from repository
                myRepository.deleteData()
                MyState()
            }
        }
    }
}

第5步:修改视图

修改视图以呈现模型提供的状态。视图不应再负责获取数据或更新状态。

class MyActivity : AppCompatActivity() {
    private val viewModel: MyViewModel by viewModels()

    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        setContentView(R.layout.activity_my)

        viewModel.state.observe(this, Observer { state ->
            // Render state in UI
            when {
                state.isLoading -> showLoading()
                state.error != null -> showError(state.error)
                state.data != null -> showData(state.data)
            }
        })
    }

    private fun showLoading() {
        // Show loading spinner or progress bar
    }

    private fun showError(error: Throwable) {
        // Show error message
    }

    private fun showData(data: Data) {
        // Show data in UI
    }
}

结论

总之,从 MVVM 迁移到 MVI 可以为您的 Android 应用带来多项好处。通过采用 MVI 架构,您可以创建更具反应性、可预测性和可测试性的应用程序,同时减少层之间的耦合并提高代码库的整体可维护性。

MVI 允许您将用户交互建模为意图,根据这些意图更新应用程序的状态,然后在 UI 中呈现该状态。这使得您可以更轻松地推断应用程序中的数据流,并更改代码,而不会引入意外的副作用。此外,MVI 有助于避免 MVVM 中的常见问题,例如“回调地狱”和“状态爆炸”。

当然,从 MVVM 迁移到 MVI 存在一些挑战,例如需要重构现有代码以及学习新的概念和模式。然而,通过仔细的规划、清晰的文档和彻底的测试,MVI 的好处可以远远超过成本。通过采用像 MVI 这样的反应式和可预测的架构,您可以创建更可靠、可扩展和可维护的 Android 应用程序,以满足用户和利益相关者的需求。

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

推荐阅读更多精彩内容