Android现代开发推荐 | Android Showcase 2.0

Android现代开发推荐 | Android Showcase 2.0

Android Showcase是一个完整的Android应用程序示例,它使用了现代的Android应用程序开发方法,集成了流行的开发工具、库和代码检查工具,以及强大的测试框架和持续集成(CI)设置。该项目的主要重点是推广模块化、可扩展、可维护和可测试的架构,并结合了最佳的软件开发实践。即使这个应用程序看起来很简单,它包含了所有关键组件,为强大的大规模应用程序打下了基础。

该项目中采用的设计原则和架构选择非常适合更大的团队和更长的应用程序生命周期。这个应用程序不仅展示了功能,而且证明了良好结构化和编写的代码如何作为可扩展和可维护软件开发项目的稳定支柱。无论是新手还是有经验的开发者,都可以从这个项目中学到很多东西。

应用简介

Android Showcase是一个展示各种音乐专辑信息的简单应用程序,它通过使用Last.fm音乐平台API动态获取数据。该应用程序包含多个特性模块,其中包括以下屏幕:

  • 专辑列表屏幕(Album list screen):显示专辑列表
  • 专辑详细信息屏幕(Album detail screen):显示所选专辑的信息
  • 个人资料屏幕(Profile screen):目前是空白(WiP)
  • 收藏夹屏幕(Favourites screen):目前是空白(WiP)

这些屏幕旨在展示应用程序设计和架构的不同方面,以及如何将它们整合到一个完整的Android应用程序中。无论您是初学者还是有经验的开发者,都可以从这个应用程序中学到很多东西。

技术栈

该项目利用 Android 生态系统中的最佳实践和许多流行的库和工具。除非有很好的理由使用非稳定的依赖项,否则大多数库都是稳定版本。

  • 技术栈
    • 100% Kotlin
      • 协程 - 执行后台操作
      • Kotlin Flow - 跨越所有应用层的数据流,包括视图。
      • Kotlin 符号处理 - 启用编译器插件
      • Kotlin 序列化 - 解析 JSON
    • Retrofit - 网络通信
    • Jetpack
      • Compose - 现代、本地 UI 工具包
      • Navigation - 应用内导航
      • Lifecycle - 在生命周期状态更改时执行操作
      • ViewModel - 以生命周期感知的方式存储和管理与 UI 相关的数据
      • Room - 存储离线缓存
    • Koin - 依赖注入(依赖检索)
    • Coil - 图片加载库
    • Lottie - 动画库
  • 现代架构
    • Clean Architecture
    • 使用 Navigation 组件 的单一活动架构
    • MVVM + MVI(表示层)
    • Android 架构组件
      (ViewModel
      , Kotlin Flow
      , Navigation)
    • Android KTX - Jetpack Kotlin 扩展
  • UI
    • 响应式 UI
    • Jetpack Compose - 现代、本地 UI 工具包(用于 Fragment)
    • View Binding - 检索 XML 视图 ID
      (仅用于 NavHostActivity)
    • Material Design 3 - 应用设计系统,提供 UI 组件
    • 主题选择
      • 深色主题 - 应用程序的深色主题(Android 10+)
      • 动态主题 - 使用生成的基于壁纸的主题(Android 12+)
  • CI
    • GitHub Actions
    • 自动 PR 验证,包括测试、linter 和第三方在线工具
  • 测试
    • 单元测试(通过 android-junit5 使用 JUnit 5)- 测试单个类
    • Konsist - 测试代码约定和架构规则
    • UI 测试(Espresso)-
      测试用户界面(WiP)
    • Mockk - 模拟框架
    • Kluent - 断言框架
  • 静态分析工具(linter)
    • Ktlint - 验证代码格式
    • Detekt - 验证代码复杂度和代码气味
    • Android Lint - 验证 Android 平台使用情况
  • Gradle
    • Gradle Kotlin DSL - 定义构建脚本
    • 自定义任务
    • Gradle 插件
      • Android Gradle - 标准 Android 插件
      • Test Logger - 格式化测试日志
      • SafeArgs - 在导航目的地之间传递数据
      • Android-junit5 - 使用 JUnit 5 与 Android
    • 版本目录 - 定义依赖关系
    • 类型安全访问器
  • GitHub 引导
    • Renovate - 自动更新依赖项
    • Stale - 自动关闭陈旧的问题和拉取请求,这些问题和拉取请求在项目中容易积累
  • 其他工具
    • Charles Proxy - 在 debug 构建中启用网络流量嗅探。

模块化架构

我们使用模块化的方式来减少大型系统的复杂性。每个模块是一个独立的构建块,具有明确的目标。我们将每个特性看作是可重用的组件,类似于microservice或私有库。

采用模块化的代码库方法有以下好处:

  • 可重用性 - 可以共享代码并从相同的基础构建多个应用程序。应用程序应该是其特性的总和,其中特性被组织为单独的模块。
  • 关注点分离 - 每个模块都有明确的API。与特性相关的类位于不同的模块中,并且没有显式的模块依赖项不能引用它们。我们严格控制着哪些内容对代码库的其他部分公开。
  • 特性可以并行开发,例如由不同的团队
  • 每个特性可以独立于其他特性进行开发
  • 构建时间更快

模块类型和模块依赖关系

下面的图表展示了项目模块(Gradle 子项目)之间的依赖关系。

应用程序有三种类型的模块:

  • app 模块 - 这是主模块。它包含将多个模块连接在一起的代码(类、依赖注入设置、NavHostActivity等)和基本应用程序配置(Retrofit 配置、所需权限设置、自定义 Application 类等)。
  • feature_x 模块 - 最常见的模块类型,包含与给定特性相关的所有代码。可以在 feature 模块之间共享一些资源或代码(当前应用程序没有此类模块)。
  • feature_base 模块 - 特性模块依赖于它们以共享公共代码。

特性模块结构

我们采用 Clean Architecture 在模块级别实现 - 每个模块都包含自己一组 Clean Architecture 层:

注意,app 模块和 library_x 模块的结构与特性模块的结构稍有不同。

每个特性模块包含非层组件和3个具有不同责任集的层。

表示层

该层最接近用户在屏幕上看到的内容。

presentation 层结合了 MVVMMVI 模式:

  • MVVM - 使用 Jetpack 的 ViewModel 封装一个 common UI state。它通过可观察的状态持有者(Kotlin Flow)公开状态。
  • MVI - action 修改 common UI state 并通过 Kotlin Flow 向视图发出新的状态。

common state 是每个视图的单一真理来源。这种解决方案源于 单向数据流 和 Redux 原则。

这种方法有助于创建一致的状态。状态通过 collectAsUiStateWithLifecycle 方法收集。Flow 集合以受生命周期管理的方式进行,因此不会浪费资源。

状态使用 Immutable 注解标记,Jetpack Compose 使用该注解启用组合优化。

组件:

  • 视图(Fragment) - 观察通用视图状态(通过 Kotlin Flow)。Compose 将由 Kotlin Flow 发出的状态转换为应用程序 UI。将用户交互传递给 ViewModel。视图很难进行测试,因此应尽可能简单。
  • ViewModel - 向视图发出(通过 Kotlin Flow)视图状态的更改,并处理用户交互(这些视图模型不仅仅是POJO 类)。
  • ViewState - 单个视图的通用状态
  • StateTimeTravelDebugger - 记录操作和视图状态转换以便进行调试。
  • NavManager - 单例,用于在 NavHostActivity 内处理所有导航事件(而不是在每个视图内单独处理)。
领域层

这是应用程序的核心层。请注意,domain 层独立于其他任何层。这样可以使领域模型和业务逻辑独立于其他层。换句话说,对其他层的更改(例如更改数据库(data 层)或屏幕 UI(presentation 层))理想情况下不会影响 domain 层的任何代码更改。

组件:

  • UseCase - 包含业务逻辑
  • DomainModel - 定义应用程序中将使用的数据的核心结构。这是应用程序数据的真理来源。
  • Repository 接口 - 保持 domain 层独立于 data layer(依赖反转)。
数据层

包装应用程序数据。为 domain 层提供数据,例如从互联网检索数据并在设备离线时将数据缓存到磁盘缓存中。

组件:

  • Repository 将数据公开给 domain 层。根据应用程序结构和外部 API 的质量,存储库还可以合并、过滤和转换数据。这些操作旨在为 domain 层创建高质量的数据源。存储库(一个或多个)负责从 Data Source 中读取并构建领域模型,并接受要写入 Data Source 的领域模型。
  • Mapper - 将 data model 映射到 domain model(使 domain 层独立于 data 层)。

该应用程序有两个 Data Sources - Retrofit(用于网络访问)和 Room(用于访问设备持久存储的本地存储)。这些数据源可以被视为一个隐式子层。每个数据源由多个类组成:

  • Retrofit Service - 定义一组 API 端点
  • Retrofit Response Model - 定义给定端点的网络对象(数据的顶层模型包含 ApiModels
  • Retrofit Api Data Model - 定义网络对象(响应模型的子对象)
  • Room Database - 持久性数据库,用于存储应用程序数据
  • Room DAO - 与存储的数据进行交互
  • Room Entity - 存储对象的定义

Retrofit API Data ModelsRoom Entities 都包含注解,因此给定的框架了解如何解析数据为对象。

数据流

下图展示了当用户与“专辑列表界面”交互时的应用程序数据流:

依赖管理

使用 Gradle 版本目录 作为集中化的依赖管理,共享第三方依赖项坐标(分组、构件、版本)跨所有模块(Gradle 项目和子项目)。

所有依赖项都存储在 settings.gradle.kts 文件中(默认位置)。Gradle 版本目录包括几个主要部分:

  • [versions] - 声明可被所有依赖项引用的版本
  • [libraries] - 声明库坐标的别名
  • [bundles] - 声明依赖捆绑包(组)
  • [plugins] - 声明 Gradle 插件依赖项

每个特性模块都依赖于 feature_base 模块,因此依赖项在不需要在每个特性模块中显式添加的情况下共享。

该项目启用了 TYPESAFE_PROJECT_ACCESSORS 实验性 Gradle 功能,以生成类型安全的访问器,以引用其他项目。

// Before
implementation(project(":feature_album"))// After
implementation(projects.featureAlbum)

灵感

以下是一些额外的资源。

快速参考表

  • Material Theme Builder - 生成动态材料主题并查看其实际效果
  • Compose Material 3 Components
    - 包含材料组件的列表
  • Core App Quality Checklist - 学习如何构建高质量应用程序
  • Android Ecosystem Cheat Sheet - 包含 200+ 最重要的工具的板
  • Kotlin Coroutines - Use Cases on Android -
    最流行的协程用途

Android 项目

其他高质量的项目将帮助您找到适用于您的项目的解决方案(随机顺序):

  • Compose Samples - 存储库包含一组个别的 Android Studio
  • Jetpack Compose Playground - 这是一个 Jetpack Compose 示例
    项目
  • Now Android - 完全使用 Kotlin 和 Jetpack Compose 构建的功能齐全的 Android 应用程序
  • WhatsApp Clone Compose - 使用 Jetpack Compose 和 Stream Chat SDK
    构建的 WhatsApp 克隆应用程序,用于帮助您了解 Android 中的 Compose
  • Iosched - Google IO 2019 和 2021 的官方 Android 应用程序
  • Android Architecture Blueprints v2 - 展示了各种 Android
    架构方法来开发 Android 应用程序
  • Github Browser Sample - 多个小项目,演示了 Android
    架构组件的使用
  • Clean Architecture Boilerplate - 用于 Android 的清晰
    架构
  • Roxie - common state 方法的坚实示例以及非常好的文档
  • Kotlin Android Template - 模板,可让您在几秒钟内创建预配置的 Android Kotlin 项目

项目地址

https://github.com/igorwojda/android-showcase

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

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

相关文章

《WebKit 技术内幕》之四(3): 资源加载和网络栈

3. 网络栈 3.1 WebKit的网络设施 WebKit的资源加载其实是交由各个移植来实现的,所以WebCore其实并没有什么特别的基础设施,每个移植的网络实现是非常不一样的。 从WebKit的代码结构中可以看出,网络部分代码的确比较少的,它们都在…

AI 编程的机会和未来:从 Copilot 到 Code Agent

大模型的快速发展带来了 AI 应用的井喷。统计 GPT 使用情况,编程远超其他成为落地最快、使用率最高的场景。如今,大量程序员已经习惯了在 AI 辅助下进行编程。数据显示,GitHub Copilot 将程序员工作效率提升了 55%,一些实验中 AI …

Autosar信息安全入门系列01-SecOC基础介绍

本文框架 1. 概述2. SecOC基本概念2.1 SecOC是什么?2.2 新鲜度值与MAC值2.3 SecOC报文格式 3. SecOC报文发送及接收逻辑3.1 SecOC报文的发送3.2 SecOC报文的接收 1. 概述 本文为Autosar通信入门系列介绍,如您对AutosarMCAL配置,通信&#xf…

【开源】基于JAVA语言的免税店商城管理系统

目录 一、摘要1.1 项目介绍1.2 项目录屏 二、系统设计2.1 功能模块设计2.2 研究方法 三、系统展示四、核心代码4.1 查询免税种类4.2 查询物品档案4.3 新增顾客4.4 新增消费记录4.5 审核免税 五、免责说明 一、摘要 1.1 项目介绍 基于JAVAVueSpringBootMySQL的免税店商城管理系…

AI嵌入式K210项目(15)-安全散列算法加速器

文章目录 前言一、什么是SHA256?实验原理 二、K210的安全散列算法加速器三、实验过程总结 前言 K210内置了丰富的加速器,包括神经网络处理器 (KPU),AES(高级加密加速器),APU 麦克风阵列语音数据加速计算处理器,现场可…

2024--Django平台开发-订单项目管理用户认证+动态菜单+权限控制(十三)

day13 订单管理项目开发 1.表结构设计 1.1 abstract类 from django.db import modelsclass ActiveBaseModel(models.Model):active models.SmallIntegerField(verbose_name"状态", default1, choices((1, "激活"), (0, "删除"),))class Meta:…

交叉编译工具 aarch64-linux-gnu-gcc 的介绍与安装

AArch64 是随 ARMv8 ISA 一起引入的 64 位架构,用于执行 A64 指令的计算机。而且在 AArch64 状态下执行的代码只能使用 A64 指令集。,而不能执行 A32 或 T32 指令。但是,与 AArch32 中不同,在64位状态下,指令可以访问 …

Java异常

异常 认识异常 异常就是代表程序出现的问题 Exception:叫异常,它代表的才是我们程序可能出现的问题,所以,我们通常会用Exception以及它的孩子来封装出现出现的问题。 运行时异常:RuntimeException及其子类&#xff…

MyBatisPlus学习笔记五-插件功能

0、插件功能 MyBatisPlus提供的内置拦截器有下面这些 1、分页插件 2、通用分页实体 3、通用分页实体-强化 需求: 在PageQuery中定义方法,将PageQuery对象转为MyBatisPlus中的Page对象在PageDTO中定义方法,将MyBatisPlus中的Page结果转为Page…

go语言(八)---- map

map的声明方式有以下三种。 package mainimport "fmt"func main() {//第一种声明方式//声明map1是一个map类型,key是String,value是Stringvar myMap1 map[string] stringif myMap1 nil {fmt.Println("myMap1 是一个空map")}//在使…

Datawhale 强化学习笔记(二)马尔可夫过程,DQN 算法

文章目录 参考马尔可夫过程DQN 算法(Deep Q-Network)如何用神经网络来近似 Q 函数如何用梯度下降的方式更新网络参数强化学习 vs 深度学习 提高训练稳定性的技巧经验回放目标网络 代码实战 DQN 算法进阶Double DQNDueling DQN 算法代码实战 参考 在线阅…

基于LDA的评论大数据的分析及主题建模

1.微博的关键词大数据采集; 已完成,待优化 2.LDA 错误1:使用了import pyLDAvis.sklearn,提示没有模块no module named pyldavis.sklearn。 默认安装 pyLDAvis3.4.1,最后降级处理,解决方式: …

docker ps -a 要求只显示自己想要的信息

在使用 docker ps -a 要求只显示这下面这几个字段,不显示其他的内容,方便查看哪些容器在运行,那些已经挂起。 例子: 我只想想显示这几个字段: CONTAINER ID STATUS NAMES docker ps -a --format "table {{.ID}…

EasyX图形化学习(三)

1.帧率: 即每秒钟界面刷新次数,下面以60帧为例: 1.数据类型 clock_t: 用来保存时间的数据类型。 2.clock( ) 函数: 用于返回程序运行的时间,无需参数。 3.例子: 先定义所需帧率: const …

20230118-【UNITY 学习】增加攀登系统

替换脚本PlayerMovement_04.cs using System.Collections; using System.Collections.Generic; using UnityEngine;public class PlayerMovement_05 : MonoBehaviour {private float moveSpeed; // 玩家移动速度public float walkSpeed 7; // 行走速度public float sprintSpee…

[笔记]深度学习入门 基于Python的理论与实现(一)

代码仓库 gitee 1. python 入门 1.5之前是python安装和基础语法, 我直接跳过了 1.5 Numpy 深度学习中经常出现数组和矩阵运算,Numpy 的数组类 numpy.array 提供了很多便捷的方法 1.5.1 导入 Numpy import numpy as np1.5.2 生成 Numpy 数组 np.array()&#xf…

C++ 设计模式之策略模式

【声明】本题目来源于卡码网(题目页面 (kamacoder.com)) 【提示:如果不想看文字介绍,可以直接跳转到C编码部分】 【设计模式大纲】 【简介】什么是策略模式(第14种模式) 策略模式是⼀种⾏为型设计模式&…

西门子WINCC常用C脚本1

1.置位,复位,取反 获取变量值:GetTagBit(可以是位也可以是字节,字,双字等具体字母不同) 设置变量值:SetTagBit 置位:SetTagBit("变量名",1) 复位&#xff…

Node+Express编写接口---前端

前端页面 vue_node_admin: 第一个以node后端,vue为前端的后台管理项目https://gitee.com/ah-ah-bao/vue_node_admin.git

java-log4j日志冲突解决

一、概述 java日志框架较多&#xff0c;其中主流的slf4j和commons-logging是日志接口&#xff0c;log4j、log4j2和logback是真正的日志实现库。 二、具体库单独使用 2.1 log4j <dependency><groupId>log4j</groupId><artifactId>log4j</artifa…