Clean架构与MVVM:助你打造优质Android应用

news2025/2/11 15:44:40



Android应用程序开发可能具有挑战性,特别是在创建可扩展和可维护的代码以适应不断变化的需求和用户需求方面。为了解决这个挑战,开发者通常依赖于软件架构模式,为代码组织和关注点分离提供了坚实的基础。在Android开发中,两种流行的架构模式是Model-View-ViewModel (MVVM) Clean架构。

这两种方法都强调模块化、可测试性和关注点分离,但它们在具体实施细节上有所不同。在本文中,我们将探讨如何结合MVVM和Clean架构来创建健壮且易于测试、扩展和维护的Android应用程序。我们还将介绍使用这种强大组合的最佳实践和实现技巧。

Model-View-ViewModel + Repository Layer

Model (data) — View (UI) — ViewModel (state & logic)

为什么使用MVVM架构?

  • 主要目标:将演示逻辑与应用的业务逻辑分离
  • 更好的可测试性
  • 遵循SOLID原则
  • 等等

Model

代表数据提供者

  • REST
  • Database
  • Preferences

View

代表

  • UI组件
  • 显示数据

Viewmodel

代表显示逻辑

  • 准备数据以供呈现
  • 使用观察者模式通知变更

怎样在没有使用“干净架构” (Clean Architecture) 的情况下实现它?

Clean Arch
每个开发人员都以学习和快速编写代码的愿望开始自己的职业生涯;然而,我们中的许多人缺乏编写整洁代码的概念。起初,我们努力学习快速提高,改进事物,却不知道它们在以后如何影响我们。我并不特别,当我初次开始作为Android开发人员工作时,也做了同样的事情。

清晰架构(Clean Architecture)的好处

  • 关注点的分离
  • 从数据模型驱动UI
  • 使功能易于更改或删除
  • 使代码更易读
  • 使用良好的实践和Jetpack库(架构组件)

分层的清晰架构(Clean Architecture)

https://developer.android.com/topic/architecture
Layer1:领域层(Domain Layer)

  • 位于用户界面和数据层之间
  • 在该项目中用于封装业务逻辑
  • 使用例可以在多个视图模型中重用
  • 定义驱动主要功能的存储库接口

Layer2:数据层(Data Layer) - 分为(本地和远程)

  • 包含业务逻辑的实现(存储库实现)
  • 从远程数据源获取数据
  • 将远程数据缓存至本地 Room 数据库

Layer3:表示/用户界面层(Presentation/UI layer)

  • 这一层是将数据显示给用户屏幕的层
  • 包含生命周期友好的视图模型,并且将代码从活动/用户界面组件中抽离出来
  • 定义了我们的架构,即 MVVM(模型视图视图模型)
  • 包含处理逻辑,如加载的状态

示例工程

源码地址如下:

https://github.com/moemaair/Lictionary
Lictionary Search

Lictionary App
代码结构如下:
code structure
我们将从将模型实体、对象或POJO类添加到数据目录开始。
POJO类
接下来是创建 Repository,它只是一个接口,请记得创建其实现,但将其放在数据目录中,而这个接口应该放在领域层以实现互操作性。
data dir
我们还没有获取任何数据,所以让我们记住,在数据层中将其分为两个部分(本地和远程)。

数据层的结构
data package

在远程中,我们将其放在数据传输对象(DTO)和接口中,使用您喜欢的HTTP客户端连接到远程API,我使用了Retrofit!

https://api.dictionaryapi.dev获取单词。

WordInfoRepo实现如下:

class WordInfoRepoImpl(
    private val api: LictionaryApi,
    private val dao:WordInfoDao
) : WordInfoRepo {

    override fun getWordInfo(word: String): Flow<Resource<List<WordInfo>>> = flow {
        emit(Resource.Loading())

        val wordInfos = dao.getWordInfos(word).map { it.toWordInfo()}
        emit(Resource.Loading(data = wordInfos))

        try {
            val remoteWordInfos = api.getWordInfo(word)
            dao.insertWordInfos(remoteWordInfos.map { it.toWordInfoEntity() })

        } catch(e: HttpException) {
            emit(Resource.Error(
                message = "Sorry pal, we couldn't find definitions for the word you were looking for.",
                data = wordInfos
            ))
        } catch(e: IOException) {
            emit(Resource.Error(
                message = "Couldn't reach server, check your internet connection.",
                data = wordInfos
            ))
        }

        val newWordInfos = dao.getWordInfos(word).map { it.toWordInfo() }
        emit(Resource.Success(newWordInfos))
    }

    override fun deleteAll(){
        dao.deleteAllWordInfos()
    }

UseCase

UseCase是代表应用程序需要执行的单个任务的业务逻辑。对于我们只是希望从API接收“words”,我们称之为“GetWordInfo”。

在Presentation层使用ViewModel

@HiltViewModel
class MainViewModel @Inject constructor(
    private val getWordInfo: GetWordInfo,
    private val repo: WordInfoRepoImpl
) : ViewModel() {

    private val _distinctWords = MutableStateFlow<List<WordInfo>>(emptyList())
    val distinctWords: StateFlow<List<WordInfo>> = _distinctWords

    var _searchQuery = mutableStateOf("")
    var searchQuery: State<String> = _searchQuery

    var checkIfDarkmode = mutableStateOf(false)

    fun setDarkmode(darkmode: Boolean) {
        checkIfDarkmode.value = darkmode
    }

    private var _state = mutableStateOf(WordInfoState())
    var state: State<WordInfoState> = _state

    private val _eventFlow = MutableSharedFlow<UIEvent>()
    val eventFlow = _eventFlow.asSharedFlow()

    private var searchJob: Job? = null

    fun onSearch(query: String) {
        _searchQuery.value = query
        searchJob?.cancel()
        searchJob = viewModelScope.launch {
            delay(500L)
            getWordInfo(query)
                .onEach { result ->
                    when (result) {
                        is Resource.Success -> {
                            _state.value = state.value.copy(
                                wordInfoItems = result.data ?: emptyList(),
                                isLoading = false
                            )
                        }
                        is Resource.Error -> {
                            _state.value = state.value.copy(
                                wordInfoItems = result.data ?: emptyList(),
                                isLoading = false
                            )
                            _eventFlow.emit(
                                UIEvent.ShowSnackbar(
                                    result.message ?: "Unknown error"
                                )
                            )
                        }
                        is Resource.Loading -> {
                            _state.value = state.value.copy(
                                wordInfoItems = result.data ?: emptyList(),
                                isLoading = true
                            )
                        }
                    }
                }.launchIn(this)
        }
    }

    fun deleteAll(){
        repo.deleteAll()
    }
    sealed class UIEvent {
        data class ShowSnackbar(val message: String): UIEvent()
    }

现在我们可以使用来自 API 的数据并根据您可能拥有的最佳 UI 设计进行自定义。

源码地址

https://github.com/moemaair/Lictionary

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/645960.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

7-基于51单片机的金属探测器检测金属报警器设计(源程序+原理图+PCB+论文)全套资料

编号: 0007 本系统采用单片机1602液晶按键比较器蜂鸣器发光二极管组合而成。 按键说明: 加键、减键。单独一个按键为复位按键 1.单片机型号: STC89C52/51、AT89C52/51、AT89S52/51 可任选&#xff0c;程序通用2.产品自带单片机上电复位电路、手动复位电路(复位按键)、晶振电路(…

Unity入门8——音效系统

一、音频文件参数面板 Force To Mono&#xff1a;多声道转单声道 Normalize&#xff1a;强制为单声道时&#xff0c;混合过程中被标准化 Load In Background&#xff1a;后台加载&#xff0c;不阻塞主线程&#xff0c;适合大音效 Ambisonic&#xff1a;立体混响声 非常适合 36…

Java数据结构之第十六章、并查集

一、并查集原理 在一些应用问题中&#xff0c;需要将n个不同的元素划分成一些不相交的集合。开始时&#xff0c;每个元素自成一个单元素集合&#xff0c;然后按一定的规律将归于同一组元素的集合合并。在此过程中要反复用到查询某一个元素归属于那个集合的运算。适合于描述这类…

【CSDN铁粉】(获取铁粉的终极秘诀)

目录 铁粉和普通粉丝有什么区别什么是CSDN中的铁粉CSDN铁粉的用处如何获得更多的铁粉如何吸引更多的铁粉总结 目录) 铁粉和普通粉丝有什么区别 在CSDN社区中&#xff0c;铁粉和普通粉丝是两个不同的等级。铁粉是指对某个特定领域或某个专业人士非常崇拜和追随的用户&#xff0…

leetcode:1822. 数组元素积的符号(python3解法)

难度&#xff1a;简单 已知函数 signFunc(x) 将会根据 x 的正负返回特定值&#xff1a; 如果 x 是正数&#xff0c;返回 1 。如果 x 是负数&#xff0c;返回 -1 。如果 x 是等于 0 &#xff0c;返回 0 。 给你一个整数数组 nums 。令 product 为数组 nums 中所有元素值的乘积。…

Compose Desktop 实战 宝可梦图鉴

Compose Desktop 实战 宝可梦图鉴 前言 阅读本文需要一定compose基础&#xff0c;如果没有请移步Jetpack Compose入门详解&#xff08;实时更新&#xff09; 接口数据来源于pokeapi 项目源代码 如果你觉得不错&#xff0c;请给我一个star&#xff0c;THKS 实现效果 闲话不…

unity制作捕鱼达人

文章目录 介绍制作水波特效制作多种ui制作不同种类鱼的动画鱼的多种移动效果制作鱼的生成点多种炮台多种子弹多种网游戏控制器声音控制器游戏存档游戏开始 介绍 水波荡漾的特效 鱼有多种运动轨迹 每隔一段时间自动收集金币 可以切换不同的炮台 升级后有不同的特效 捕捉到普通鱼…

Salesforce开发人员如何利用生成式AI?

AI浪潮来袭&#xff0c;技术和产品的新消息奔涌而来&#xff0c;开发者们的工作模式正在经历巨变。ChatGPT的出现&#xff0c;让问题的解法更有想象力&#xff0c;也让敲下一行代码、发布一款产品变得更容易。 AI可以帮助指导开发过程中的错误&#xff0c;并改进解决方案&…

ChatGPT/InstructGPT详解

前言 GPT系列是OpenAI的一系列预训练文章&#xff0c;GPT的全称是Generative Pre-Trained Transformer&#xff0c;顾名思义&#xff0c;GPT的目的就是通过Transformer为基础模型&#xff0c;使用预训练技术得到通用的文本模型。目前已经公布论文的有文本预训练GPT-1&#xff…

【iM群发部署,苹果推是什么】苹果日历推,配置 iOS 设备:您可以创建自定义配置文件

Apple Configurator 是苹果官方提供的一款工具&#xff0c;主要面向企业和教育机构&#xff0c;用于集中管理和配置多个 iOS 设备。 使用 Apple Configurator&#xff0c;您可以在 PC 端进行以下操作&#xff1a; 配置 iOS 设备&#xff1a;您可以创建自定义配置文件&#xf…

经典文献阅读之--A Review of Motion Planning(轨迹规划回顾)

0. 简介 对于自动驾驶以及机器人而言&#xff0c;除了SLAM以外&#xff0c;另一个比较重要的部分就是轨迹规划了。而最近作者看到了几篇比较好的文章&#xff0c;分别为《A Review of Motion Planning Techniques for Automated Vehicle》、《A review of motion planning alg…

《现代中学生》初中版期刊简介及投稿邮箱

《现代中学生》初中版期刊简介&#xff1a; 《现代中学生》现代中学生初中版 主管单位 吉林省教育厅 主办单位 吉林教育杂志社 国际刊号ISSN&#xff1a;1009-5748&#xff1b;国内刊号CN&#xff1a;22-1046/G4&#xff1b;邮发代号&#xff1a;12-52 出版周期&#xff1…

关于MySQL数据库的基本概念

MySQL数据库初体验 一、数据库的基本概念二、 数据库的发展三、主流的数据库介绍四、 关系数据库五、非关系数据库 一、数据库的基本概念 1. 数据 (Data) 数据就是描述事物的符号记录。包括数字&#xff0c;文字、图形、图像、声音、档案记录等。以“记录”形式按统一的格式进…

三勾点餐新增功能特价团购

点击后台-插件-特价团购&#xff0c;展示下方界面 ☆ 团购订单 所有团购的订单将在此处展示订单搜索&#xff1a;填写任意项点击“查询即可订单详情&#xff1a;可以看到单个订单的基本信息、门店信息、团购信息、付款信息、券码信息订单核销&#xff1a;前端核销&#xff0c;…

MySQL性能深度优化

这里的深度优化是指&#xff0c;除了建索引、左匹配索引等等其他的优化手段。 文章涉及到操作系统连接数、IO、Mysql本身的某些参数设置&#xff0c;值得记录下来。 文章目录 一.数据库服务器配置二.CPU的优化三.内存的优化四.IO的优化五.连接的优化六.数据一致性的优化原文链…

scDesign3:多模态单细胞和空间组学数据生成

scDesign3是一个统计模拟器&#xff0c;通过从真实数据中学习可解释的参数&#xff0c;生成真实的单细胞和空间组学数据&#xff0c;包括各种细胞状态、实验设计和特征模态。使用单细胞和空间组学数据的统一概率模型&#xff0c;scDesign3可以推断出具有生物学意义的参数&#…

DC降压电源模块 输入8-32V输出电流50A/1V-12V 恒流恒压可调模块

【产品参数】 [] 非隔离同步降压(BUCK)电源模块 [] 宽输入DC6-32V&#xff0c;输出DC0.6-12.2V [] 峰值效率>96% [] 过流保护 短路保护自恢复 [] 远程ON/OFF [] 过温保护、高稳压精度、动态响应快 [] 输出限流指示灯 [] 输出电压电流可调 [] 低纹波、低静态电流 […

Kafka学习--3、Kafka Broker、节点服役和退役、Kafka 副本、Leader 选举流程、故障处理

1、Kafka Broker 1.1 Kafka Broker工作流程 1.1.1 Zookeeper储存的Kafka信息 &#xff08;1&#xff09;启动Zookeeper集群、再启动Kafka集群&#xff0c;然后启动Zookeeper客户端 &#xff08;2&#xff09;通过ls命令可以查看kafka相关信息。 1.1.2 Kafka Broker总体工作…

【冷冻电镜】IMOD使用教程(Etomo tutorial)

参考教程&#xff1a; Etomo Tuturial for IMOD version 4.11 1. Initial Setup 本教程提供了一个小双轴示例数据集和Etomo的分布指南&#xff0c;更详细的内容参考Tomography Guide。该版本使用1k*1k的图像而不是压缩版本。imodhelp命令可以打开帮助界面&#xff0c;查看各种…

自定义修改Typora原生默认github风格样式

使用typora的时候&#xff0c;想要自定义一些颜色、字体&#xff0c;或者修改一些设置&#xff0c;这个时候需要修改或者自己编写css文件。 修改涉及的样式&#xff1a; ① 目录 ② 块应用 我还是比较喜欢原生自带的默认样式&#xff08;github样式&#xff09;&#xff0c; 但…