当前位置: 首页 > news >正文

Redux 与 MVI:Android 应用的对比

Redux 与 MVI:Android 应用的对比

在为 Android 应用选择合适的状态管理架构时可能会感到困惑。在这个领域中,有两种流行的选择是 Redux 和 MVI(Model-View-Intent)。两者都有各自的优缺点,因此在深入研究之前了解它们的区别至关重要。

本指南将深入探讨 Redux 和 MVI 的核心原则,突出它们在 Android 开发中的关键区别。此外,我将提供一些有用的资源链接,这些资源提供了更深入的见解和实用的实现示例。

Redux

  1. 集中式状态存储:Redux 将所有应用程序状态存储在一个称为“store”的单个不可变数据结构中。这种集中式方法使状态转换可预测且易于调试。
  2. 单向数据流:行为(Actions)代表应用程序中的事件,是修改状态的唯一方式。行为被发送到存储库,触发减速器(Reducers)根据纯函数更新状态。这种单向流使得对状态变化的推理更加容易。
  3. 中间件处理副作用:虽然 Redux 专注于管理纯状态,但中间件函数可以处理诸如网络调用或本地存储更新等副作用。这种关注点分离保持了核心 Redux 逻辑的清晰度。
  4. 解耦的架构:MVI 将 UI(View)与业务逻辑(Model)和用户交互(Intent)分离。这种模块化促进了代码的可重用性和可测试性。
  5. 响应式状态更新:模型根据意图(Intent)发出新状态,通过响应式绑定机制自动更新视图。这消除了显式状态管理操作的需要。
  6. 不可变数据模型:与 Redux 类似,MVI 强调使用不可变数据结构来实现模型,确保可预测的状态变化和更简单的推理。

MVI

  • 解耦的架构:MVI 将用户界面(View)与业务逻辑(Model)和用户交互(Intent)分离开来。这种模块化促进了代码的可重用性和可测试性。
  • 响应式状态更新:模型通过发出新的状态来响应意图,这通过一种响应式绑定机制自动更新视图。这消除了需要显式状态管理操作的必要性。
  • 不可变数据模型:与 Redux 类似,MVI 强调使用不可变数据结构来构建模型,确保状态变化可预测且更容易推理。

Android 特定考虑因素

  1. 库和框架:Redux 和 MVI 都有专门的 Android 库和框架,例如 redux-kotlin-android 和 arkivia-mvi。这些库简化了与 Android 组件的集成,并提供了管理状态和副作用的有用工具。
  2. 测试:这两种架构都有成熟的测试方法。对于 Redux,像 redux-mock-store 这样的测试框架可以实现高效的单元测试和集成测试。MVI 的响应式特性通常通过使状态更加显式来简化测试编写。

示例代码

Redux 示例代码

Action Types 定义
sealed class ActionType {object IncrementCounter : ActionType()object DecrementCounter : ActionType()
}
Action Creator 函数
fun incrementCounter(): ActionType = ActionType.IncrementCounter
fun decrementCounter(): ActionType = ActionType.DecrementCounter
Reducer 函数
fun reducer(state: Int, action: ActionType): Int {return when (action) {is ActionType.IncrementCounter -> state + 1is ActionType.DecrementCounter -> state - 1}
}
Store 创建与初始化
class Store(private val reducer: (Int, ActionType) -> Int) {private var state: Int = 0private val listeners: MutableList<() -> Unit> = mutableListOf()fun getState(): Int = statefun dispatch(action: ActionType) {state = reducer(state, action)listeners.forEach { it.invoke() }}fun subscribe(listener: () -> Unit) {listeners.add(listener)}
}
使用示例
fun main() {val store = Store(::reducer)val listener: () -> Unit = { println("Current counter value: ${store.getState()}") }store.subscribe(listener)store.dispatch(incrementCounter())store.dispatch(incrementCounter())store.dispatch(decrementCounter())
}

MVI 示例代码

Model 定义
data class CounterModel(val count: Int)
Intent 类型定义
sealed class CounterIntent {object Increment : CounterIntent()object Decrement : CounterIntent()
}
ViewModel 创建与初始化
class CounterViewModel : ViewModel() {private val _counterState = MutableLiveData<CounterModel>()val counterState: LiveData<CounterModel>get() = _counterStateinit {_counterState.value = CounterModel(0)}fun processIntent(intent: CounterIntent) {val currentCount = _counterState.value?.count ?: 0when (intent) {is CounterIntent.Increment -> _counterState.value = CounterModel(currentCount + 1)is CounterIntent.Decrement -> _counterState.value = CounterModel(currentCount - 1)}}
}
使用示例
fun main() {val viewModel = CounterViewModel()val observer = Observer<CounterModel> { counterModel ->println("Current counter value: ${counterModel.count}")}viewModel.counterState.observeForever(observer)viewModel.processIntent(CounterIntent.Increment)viewModel.processIntent(CounterIntent.Increment)viewModel.processIntent(CounterIntent.Decrement)
}

注意:

  • Redux 示例中的 Store 是手动实现的简化版本,而在实际应用中通常会使用第三方库来管理 Redux Store。
  • MVI 示例中使用了 Android 架构组件的 ViewModel 和 LiveData 来实现单向数据流。

有用的资源

Redux

  • Redux 文档:https://redux.js.org/
  • Kotlin Redux 教程:https://www.youtube.com/watch?v=BUAxqiGrKOc
  • Android Redux 库:https://github.com/reduxkotlin/redux-kotlin

MVI

  • MVI 文档:https://github.com/adidas/mvi
  • Arkivia-MVI 库:https://github.com/badoo/MVICore
  • MVI vs. Redux for Android:https://medium.com/@chessmani/yup-by-the-way-mvi-is-really-no-different-from-redux-its-just-a-different-name-which-i-wish-a3f3fe334fd9

结论

选择 Redux 还是 MVI 取决于您的特定需求和偏好。在做出决定时考虑诸如项目复杂性、开发人员经验和所需的模块化水平等

相关文章:

Redux 与 MVI:Android 应用的对比

Redux 与 MVI&#xff1a;Android 应用的对比 在为 Android 应用选择合适的状态管理架构时可能会感到困惑。在这个领域中&#xff0c;有两种流行的选择是 Redux 和 MVI&#xff08;Model-View-Intent&#xff09;。两者都有各自的优缺点&#xff0c;因此在深入研究之前了解它们…...

《MySQL是怎样运行的》读书笔记(三) B+树索引

前言 从前面数据存储结构中我们已经知道了页和记录的关系示意图: 其中页a、页b、页c ... 页n 这些页可以不在物理结构上相连&#xff0c;只要通过双向链表相关联即可。 在正式介绍索引之前&#xff0c;我们需要了解一下没有索引的时候是怎么查找记录的。下边先只讨论搜索条件…...

微信小程序基础工作模板

1.轮播图 点击跳转官方文档 简单例子 <!-- 顶部轮播图 --> <swiper indicator-dots"true" class"banner" autoplay"true" interval"2000"><swiper-item><image src"../../images/轮播图1.jpg" >…...

简单说一下STL中的map容器的特点、底层实现和应用场景【面试】

特点&#xff1a; 基于红黑树&#xff1a;std::map利用红黑树的自平衡特性&#xff0c;确保操作的平衡性。有序容器&#xff1a;元素根据键的顺序自动排序&#xff0c;排序依据是预定义的键比较函数。唯一键值&#xff1a;容器保证每个键的唯一性&#xff0c;不允许重复键存在…...

Ubuntu22.04之有道词典无法画词翻译替代方案(二百四十九)

简介&#xff1a; CSDN博客专家&#xff0c;专注Android/Linux系统&#xff0c;分享多mic语音方案、音视频、编解码等技术&#xff0c;与大家一起成长&#xff01; 优质专栏&#xff1a;Audio工程师进阶系列【原创干货持续更新中……】&#x1f680; 优质专栏&#xff1a;多媒…...

AnythingLLM 的 Docker 使用

AnythingLLM是使用大语言模型LLM的一站式简便框架。官网的介绍如下&#xff1a; AnythingLLM is the easiest to use, all-in-one AI application that can do RAG, AI Agents, and much more with no code or infrastructure headaches. 1. 使用官方docker 最方便的方法是使…...

数组还可以这样用!常用但不为人知的应用场景

哈喽&#xff0c;各位小伙伴们&#xff0c;你们好呀&#xff0c;我是喵手。运营社区&#xff1a;C站/掘金/腾讯云&#xff1b;欢迎大家常来逛逛 今天我要给大家分享一些自己日常学习到的一些知识点&#xff0c;并以文字的形式跟大家一起交流&#xff0c;互相学习&#xff0c;一…...

C++模板元编程:编译时的魔法

1. 引言 在C的世界中&#xff0c;模板元编程是一种在编译时执行计算的强大技术。它允许开发者编写高度灵活和高效的代码&#xff0c;这些代码可以在不牺牲性能的前提下&#xff0c;根据类型和值的不同而变化。本文将深入探讨模板元编程的奥秘&#xff0c;并展示如何在现代C开发…...

SQL进阶day10————多表查询

目录 1嵌套子查询 1.1月均完成试卷数不小于3的用户爱作答的类别 1.2月均完成试卷数不小于3的用户爱作答的类别 ​编辑1.3 作答试卷得分大于过80的人的用户等级分布 2合并查询 2.1每个题目和每份试卷被作答的人数和次数 2.2分别满足两个活动的人 3连接查询 3.1满足条件…...

debug调试_以Pycharm为例

文章目录 作用步骤打断点调试调试窗口 作用 主要是检查逻辑错误&#xff0c;而非语法错误。 步骤 打断点 在需要调试的代码行前打断点&#xff0c;执行后会停顿在断点位置&#xff08;不运行&#xff09; 调试 右键“debug”&#xff0c;或者直接点击右上角的小虫子 调试…...

wms第三方海外仓系统:如何为中小型海外仓注入新活力

对于中小型海外仓来说&#xff0c;想在大型集团海外仓同台竞争中获得优胜&#xff0c;提升其管理效率是非常关键的一环。 我们所熟知的wms系统&#xff0c;也就是第三方成熟海外仓系统&#xff0c;正是这些海外仓企业提升管理水平、降低成本的重要工具。 1、wms第三方海外仓系…...

html是什么?http是什么?

html Html是什么&#xff1f;http是什么&#xff1f; Html 超文本标记语言&#xff1b;负责网页的架构&#xff1b; http(&#xff08;HyperText Transfer Protocol&#xff09;超文本传输协议&#xff1b; https&#xff08;全称&#xff1a;Hypertext Transfer Protocol …...

L1-007 念数字js实现

异步解法 const readline require("readline"); const rl readline.createInterface({input: process.stdin,output: process.stdout, }); const input_arr [];//储存数据 rl.on(line, function (line) {input_arr.push(line); } ); rl.on(close, function () {/…...

Perl 运算符

Perl 运算符 Perl 是一种功能强大的编程语言&#xff0c;广泛应用于系统管理、网络编程、GUI 创建、数据库访问等众多领域。Perl 的语法灵活&#xff0c;支持多种编程范式&#xff0c;包括过程式、面向对象和函数式编程。在 Perl 中&#xff0c;运算符扮演着重要的角色&#x…...

语法04 C++ 标准输入语句

标准输入 使用格式&#xff1a;cin >> 输入的意思就是把一个值放到变量里面去&#xff0c;也就是变量的赋值&#xff0c;这个值是由我们自己输入的。 (注意:输入变量前要先定义&#xff0c;输入完之后要按Enter键。) 输入多个变量&#xff0c;与输出类似&#xff0c;…...

python数据分析--- ch6-7 python容器类型的数据及字符串

python数据分析---ch6-7 python容器类型的数据及字符串 1. Ch6--容器类型的数据1.1 序列1.1.1 序列的索引操作1.1.2 加和乘操作1.1.3 切片操作1.1.4 成员测试 1.2 列表1.2.1 创建列表1.2.2 追加元素1.2.3 插入元素1.2.4 替换元素1.2.5 删除元素1.2.6 列表排序&#xff08;1&…...

【Linux取经路】守护进程

文章目录 一、前台进程和后台进程二、Linux 的进程间关系三、setsid——将当前进程设置为守护进程四、daemon——设置为守护进程五、结语 一、前台进程和后台进程 Linux 中每一次用户登录都是一个 session&#xff0c;一个 session 中只能有一个前台进程在运行&#xff0c;键盘…...

Nginx之文件下载服务器

1.概述 在对外分享文件时&#xff0c;利用Nginx搭建一个简单的下 载文件管理服务器&#xff0c;文件分享就会变得非常方便。利 用Nginx的诸多内置指令可实现自动生成下载文件列表 页、限制下载带宽等功能。配置样例如下&#xff1a; server {listen 8080;server_name localhos…...

OpenCV学习(4.11) OpenCV中的图像转换

1. 目标 在本节中&#xff0c;我们将学习 使用OpenCV查找图像的傅立叶变换利用Numpy中可用的FFT功能傅立叶变换的一些应用我们将看到以下函数&#xff1a;**cv.dft()** &#xff0c;**cv.idft()** 等 理论 傅立叶变换用于分析各种滤波器的频率特性。对于图像&#xff0c;使用…...

2024.6.13每日一题

LeetCode 子序列最大优雅度 题目链接&#xff1a;2813. 子序列最大优雅度 - 力扣&#xff08;LeetCode&#xff09; 题目描述 给你一个长度为 n 的二维整数数组 items 和一个整数 k 。 items[i] [profiti, categoryi]&#xff0c;其中 profiti 和 categoryi 分别表示第 i…...

Golang 面试经典题:map 的 key 可以是什么类型?哪些不可以?

Golang 面试经典题&#xff1a;map 的 key 可以是什么类型&#xff1f;哪些不可以&#xff1f; 在 Golang 的面试中&#xff0c;map 类型的使用是一个常见的考点&#xff0c;其中对 key 类型的合法性 是一道常被提及的基础却很容易被忽视的问题。本文将带你深入理解 Golang 中…...

FastAPI 教程:从入门到实践

FastAPI 是一个现代、快速&#xff08;高性能&#xff09;的 Web 框架&#xff0c;用于构建 API&#xff0c;支持 Python 3.6。它基于标准 Python 类型提示&#xff0c;易于学习且功能强大。以下是一个完整的 FastAPI 入门教程&#xff0c;涵盖从环境搭建到创建并运行一个简单的…...

dedecms 织梦自定义表单留言增加ajax验证码功能

增加ajax功能模块&#xff0c;用户不点击提交按钮&#xff0c;只要输入框失去焦点&#xff0c;就会提前提示验证码是否正确。 一&#xff0c;模板上增加验证码 <input name"vdcode"id"vdcode" placeholder"请输入验证码" type"text&quo…...

Java - Mysql数据类型对应

Mysql数据类型java数据类型备注整型INT/INTEGERint / java.lang.Integer–BIGINTlong/java.lang.Long–––浮点型FLOATfloat/java.lang.FloatDOUBLEdouble/java.lang.Double–DECIMAL/NUMERICjava.math.BigDecimal字符串型CHARjava.lang.String固定长度字符串VARCHARjava.lang…...

macOS多出来了:Google云端硬盘、YouTube、表格、幻灯片、Gmail、Google文档等应用

文章目录 问题现象问题原因解决办法 问题现象 macOS启动台&#xff08;Launchpad&#xff09;多出来了&#xff1a;Google云端硬盘、YouTube、表格、幻灯片、Gmail、Google文档等应用。 问题原因 很明显&#xff0c;都是Google家的办公全家桶。这些应用并不是通过独立安装的…...

Java多线程实现之Callable接口深度解析

Java多线程实现之Callable接口深度解析 一、Callable接口概述1.1 接口定义1.2 与Runnable接口的对比1.3 Future接口与FutureTask类 二、Callable接口的基本使用方法2.1 传统方式实现Callable接口2.2 使用Lambda表达式简化Callable实现2.3 使用FutureTask类执行Callable任务 三、…...

C# 类和继承(抽象类)

抽象类 抽象类是指设计为被继承的类。抽象类只能被用作其他类的基类。 不能创建抽象类的实例。抽象类使用abstract修饰符声明。 抽象类可以包含抽象成员或普通的非抽象成员。抽象类的成员可以是抽象成员和普通带 实现的成员的任意组合。抽象类自己可以派生自另一个抽象类。例…...

在鸿蒙HarmonyOS 5中使用DevEco Studio实现录音机应用

1. 项目配置与权限设置 1.1 配置module.json5 {"module": {"requestPermissions": [{"name": "ohos.permission.MICROPHONE","reason": "录音需要麦克风权限"},{"name": "ohos.permission.WRITE…...

如何在网页里填写 PDF 表格?

有时候&#xff0c;你可能希望用户能在你的网站上填写 PDF 表单。然而&#xff0c;这件事并不简单&#xff0c;因为 PDF 并不是一种原生的网页格式。虽然浏览器可以显示 PDF 文件&#xff0c;但原生并不支持编辑或填写它们。更糟的是&#xff0c;如果你想收集表单数据&#xff…...

云原生安全实战:API网关Kong的鉴权与限流详解

&#x1f525;「炎码工坊」技术弹药已装填&#xff01; 点击关注 → 解锁工业级干货【工具实测|项目避坑|源码燃烧指南】 一、基础概念 1. API网关&#xff08;API Gateway&#xff09; API网关是微服务架构中的核心组件&#xff0c;负责统一管理所有API的流量入口。它像一座…...