Android Showcase是一个完整的Android应用程序示例,它使用了现代的Android应用程序开发方法,集成了流行的开发工具、库和代码检查工具,以及强大的测试框架和持续集成(CI)设置。该项目的主要重点是推广模块化、可扩展、可维护和可测试的架构,并结合了最佳的软件开发实践。即使这个应用程序看起来很简单,它包含了所有关键组件,为强大的大规模应用程序打下了基础。
该项目中采用的设计原则和架构选择非常适合更大的团队和更长的应用程序生命周期。这个应用程序不仅展示了功能,而且证明了良好结构化和编写的代码如何作为可扩展和可维护软件开发项目的稳定支柱。无论是新手还是有经验的开发者,都可以从这个项目中学到很多东西。
Android Showcase是一个展示各种音乐专辑信息的简单应用程序,它通过使用Last.fm音乐平台API动态获取数据。该应用程序包含多个特性模块,其中包括以下屏幕:
这些屏幕旨在展示应用程序设计和架构的不同方面,以及如何将它们整合到一个完整的Android应用程序中。无论您是初学者还是有经验的开发者,都可以从这个应用程序中学到很多东西。
该项目利用 Android 生态系统中的最佳实践和许多流行的库和工具。除非有很好的理由使用非稳定的依赖项,否则大多数库都是稳定版本。
debug
构建中启用网络流量嗅探。我们使用模块化的方式来减少大型系统的复杂性。每个模块是一个独立的构建块,具有明确的目标。我们将每个特性看作是可重用的组件,类似于microservice或私有库。
采用模块化的代码库方法有以下好处:
下面的图表展示了项目模块(Gradle 子项目)之间的依赖关系。
应用程序有三种类型的模块:
app
模块 - 这是主模块。它包含将多个模块连接在一起的代码(类、依赖注入设置、NavHostActivity
等)和基本应用程序配置(Retrofit 配置、所需权限设置、自定义 Application
类等)。feature_x
模块 - 最常见的模块类型,包含与给定特性相关的所有代码。可以在 feature
模块之间共享一些资源或代码(当前应用程序没有此类模块)。feature_base
模块 - 特性模块依赖于它们以共享公共代码。我们采用 Clean Architecture
在模块级别实现 - 每个模块都包含自己一组 Clean Architecture 层:
注意,
app
模块和library_x
模块的结构与特性模块的结构稍有不同。
每个特性模块包含非层组件和3个具有不同责任集的层。
该层最接近用户在屏幕上看到的内容。
presentation
层结合了 MVVM
和 MVI
模式:
MVVM
- 使用 Jetpack 的 ViewModel
封装一个 common UI state
。它通过可观察的状态持有者(Kotlin Flow
)公开状态。MVI
- action
修改 common UI state
并通过 Kotlin Flow
向视图发出新的状态。
common state
是每个视图的单一真理来源。这种解决方案源于 单向数据流 和 Redux 原则。
这种方法有助于创建一致的状态。状态通过 collectAsUiStateWithLifecycle
方法收集。Flow 集合以受生命周期管理的方式进行,因此不会浪费资源。
状态使用 Immutable 注解标记,Jetpack Compose 使用该注解启用组合优化。
组件:
Kotlin Flow
)。Compose 将由 Kotlin Flow 发出的状态转换为应用程序 UI。将用户交互传递给 ViewModel
。视图很难进行测试,因此应尽可能简单。Kotlin Flow
)视图状态的更改,并处理用户交互(这些视图模型不仅仅是POJO 类)。NavHostActivity
内处理所有导航事件(而不是在每个视图内单独处理)。这是应用程序的核心层。请注意,domain
层独立于其他任何层。这样可以使领域模型和业务逻辑独立于其他层。换句话说,对其他层的更改(例如更改数据库(data
层)或屏幕 UI(presentation
层))理想情况下不会影响 domain
层的任何代码更改。
组件:
domain
层独立于 data layer
(依赖反转)。包装应用程序数据。为 domain
层提供数据,例如从互联网检索数据并在设备离线时将数据缓存到磁盘缓存中。
组件:
domain
层。根据应用程序结构和外部 API 的质量,存储库还可以合并、过滤和转换数据。这些操作旨在为 domain
层创建高质量的数据源。存储库(一个或多个)负责从 Data Source
中读取并构建领域模型,并接受要写入 Data Source
的领域模型。data model
映射到 domain model
(使 domain
层独立于 data
层)。该应用程序有两个 Data Sources
- Retrofit
(用于网络访问)和 Room
(用于访问设备持久存储的本地存储)。这些数据源可以被视为一个隐式子层。每个数据源由多个类组成:
ApiModels
)Retrofit API Data Models
和 Room 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)
以下是一些额外的资源。
其他高质量的项目将帮助您找到适用于您的项目的解决方案(随机顺序):
common state
方法的坚实示例以及非常好的文档https://github.com/igorwojda/android-showcase