Android 中的 ViewModel详解
在 Android 开发中,ViewModel 是 Jetpack 架构组件的核心成员之一,专为管理与界面相关的数据而设计。它通过生命周期感知能力,确保数据在配置变更(如屏幕旋转)时持久存在,并将数据逻辑与 UI 控制器(Activity/Fragment)解耦,显著提升代码的可维护性和可测试性。以下从核心概念、实现原理、使用示例及最佳实践等方面展开详细说明。
一、ViewModel 的核心作用
- 数据持久化当 Activity 或 Fragment 因配置变更(如屏幕旋转)重建时,ViewModel 会保留数据,避免重复加载。例如,网络请求结果或复杂计算结果可存储在 ViewModel 中,无需通过onSaveInstanceState手动序列化。
- 解耦 UI 与数据逻辑ViewModel 作为数据持有者,负责处理业务逻辑和数据转换,而 UI 控制器仅关注数据展示和用户交互。例如,在列表页面中,ViewModel 可封装数据获取逻辑,UI 层只需观察数据变化并更新界面。
- 生命周期安全ViewModel 的生命周期与 UI 控制器绑定,但不受配置变更影响。当 Activity/Fragment 彻底销毁时(如用户退出应用),ViewModel 会自动清理资源,避免内存泄漏。
- 跨组件通信同一作用域(如 Activity)内的多个 Fragment 可共享同一个 ViewModel 实例,实现数据共享。例如,主 Fragment 和详情 Fragment 通过共享 ViewModel 同步数据状态。
二、实现原理与关键机制
1. 生命周期管理
- ViewModelStore:每个 Activity/Fragment 持有一个ViewModelStore,用于存储其关联的 ViewModel 实例。配置变更时,ViewModelStore被保留,新创建的 UI 控制器可直接复用原 ViewModel。
- ViewModelProvider:通过工厂模式创建 ViewModel 实例。默认工厂使用反射生成实例,自定义工厂可注入依赖(如 Repository)。
2. 数据观察与更新
- LiveData 集成:ViewModel 常与 LiveData 结合,实现数据的响应式更新。LiveData 具有生命周期感知能力,仅在 UI 组件活跃时通知数据变化,避免空指针异常。
- SavedStateHandle:用于保存 ViewModel 的临时状态,即使进程被系统杀死后也能恢复。例如,表单输入或滚动位置可通过SavedStateHandle持久化。
三、使用示例:计数器应用
1. 创建 ViewModel
class CounterViewModel : ViewModel() { private val _counter = MutableLiveData(0) val counter: LiveData<Int> = _counter fun increment() { _counter.value = _counter.value?.plus(1) } fun decrement() { _counter.value = _counter.value?.minus(1) } } |
2. 在 Activity 中使用
class MainActivity : AppCompatActivity() { private lateinit var binding: ActivityMainBinding private val viewModel: CounterViewModel by viewModels() override fun onCreate(savedInstanceState: Bundle?) { super.onCreate(savedInstanceState) binding = DataBindingUtil.setContentView(this, R.layout.activity_main) binding.viewModel = viewModel binding.lifecycleOwner = this // 关联生命周期 viewModel.counter.observe(this) { count -> binding.counterText.text = count.toString() } binding.incrementButton.setOnClickListener { viewModel.increment() } binding.decrementButton.setOnClickListener { viewModel.decrement() } } } |
3. 布局文件(使用 DataBinding)
<layout xmlns:android="http://schemas.android.com/apk/res/android"> <data> <variable name="viewModel" type="com.example.viewmodeldemo.CounterViewModel" /> </data> <LinearLayout android:layout_width="match_parent" android:layout_height="match_parent" android:orientation="vertical"> <TextView android:id="@+id/counterText" android:layout_width="wrap_content" android:layout_height="wrap_content" android:text="@{viewModel.counter.toString()}" /> <Button android:id="@+id/incrementButton" android:layout_width="wrap_content" android:layout_height="wrap_content" android:text="+" android:onClick="@{() -> viewModel.increment()}" /> <Button android:id="@+id/decrementButton" android:layout_width="wrap_content" android:layout_height="wrap_content" android:text="-" android:onClick="@{() -> viewModel.decrement()}" /> </LinearLayout> </layout> |
四、高级用法:与 Room 和 Retrofit 集成
1. 架构设计
采用 MVVM 模式,通过 Repository 层封装数据来源(网络 / 本地数据库),ViewModel 负责协调数据处理。
2. ViewModel 与 Repository 交互
class UserViewModel @Inject constructor( private val userRepository: UserRepository ) : ViewModel() { val users: LiveData<List<User>> = userRepository.getUsers() fun fetchUsers() { viewModelScope.launch { userRepository.fetchAndSaveUsers() } } } |
3. Repository 层实现
class UserRepository @Inject constructor( private val api: UserApi, private val userDao: UserDao ) { fun getUsers(): LiveData<List<User>> { return userDao.getAllUsers() } suspend fun fetchAndSaveUsers() { val users = api.getUsers() userDao.insertAll(users) } } |
4. 网络请求与数据库操作
- Retrofit 接口:
interface UserApi { @GET("users") suspend fun getUsers(): List<User> } |
- Room 实体与 DAO:
@Entity(tableName = "users") data class User( @PrimaryKey val id: Long, val name: String, val email: String ) @Dao interface UserDao { @Query("SELECT * FROM users") fun getAllUsers(): LiveData<List<User>> @Insert(onConflict = REPLACE) suspend fun insertAll(users: List<User>) } |
五、最佳实践与注意事项
- 避免持有 ContextViewModel 不应直接引用 Activity/Fragment 的 Context,以免引发内存泄漏。若需 Context,可继承AndroidViewModel并通过构造函数注入Application实例。
- 数据不可变性通过 LiveData 暴露数据时,应返回不可变类型(如LiveData而非MutableLiveData),防止外部直接修改数据。
- 使用 SavedStateHandle对于需跨进程保留的状态(如搜索条件),使用SavedStateHandle存储。在 ViewModel 构造函数中注入SavedStateHandle,并通过getLiveData方法观察数据变化。
- 单元测试ViewModel 应独立于 UI 进行测试。例如,使用ViewModelTest类验证业务逻辑,模拟依赖对象(如 Repository)以隔离测试。
- 结合 DataBinding通过 DataBinding 将 UI 控件与 ViewModel 直接绑定,减少样板代码。设置lifecycleOwner确保数据更新与 UI 生命周期同步。
六、总结
ViewModel 是 Android 开发中管理 UI 数据的核心工具,其生命周期感知能力和数据持久化特性显著提升了应用的稳定性和可维护性。通过结合 LiveData、Room、Retrofit 等组件,开发者可构建高效、可扩展的架构。遵循最佳实践(如避免持有 Context、使用 Repository 模式)能进一步优化代码质量,降低维护成本。无论是简单的计数器应用还是复杂的数据驱动界面,ViewModel 都是实现清晰架构的关键组件。
相关文章:
Android 中的 ViewModel详解
在 Android 开发中,ViewModel 是 Jetpack 架构组件的核心成员之一,专为管理与界面相关的数据而设计。它通过生命周期感知能力,确保数据在配置变更(如屏幕旋转)时持久存在,并将数据逻辑与 UI 控制器…...

Java集合框架与三层架构实战指南:从基础到企业级应用
一、集合框架深度解析 1. List集合的武林争霸 ArrayList: 数组结构:内存连续,查询效率O(1) 扩容机制:默认扩容1.5倍(源码示例) private void grow(int minCapacity) {int oldCapacity elementData.len…...

6个月Python学习计划 Day 2 - 条件判断、用户输入、格式化输出
6个月Python学习计划:从入门到AI实战(前端开发者进阶指南) Python 基础入门 & 开发环境搭建 🎯 今日目标 学会使用 input() 获取用户输入掌握 if/else/elif 条件判断语法熟悉格式化输出方式:f-string、format() …...
使用docker容器部署Elasticsearch和Kibana
简介:(Elasticsearch) elasticsearch简称Es, 是位于Elastic Stack核心的分布式搜索和分析引擎。它为所有类型的数据提供近乎实时的搜索和分析。无论您拥有机构化或非结构化的文本、数字数据还是地理空间数据,Es都能以支持快速搜索…...
批量处理合并拆分pdf功能 OCR 准确率高 免费开源
各位 PDF 编辑小白们,今天咱来唠唠 PDFXEdit10_Portable 这款软件。 先说说它的核心功能和适用场景。这玩意儿是个便携式的 PDF 编辑工具,不用安装就能直接用,能改 PDF 里的文本、图片,还能批注、调整格式,老方便了。…...
Unity—lua基础语法
Lua 语言执行方式 编译型语言:代码在运行前需要使用编译器,先将程序源代码编译为可执行文件,再执行 C/C Java C# Go Objective-C 解释型语言(脚本语言) 需要提前安装编译语言解析器,运行时使用解析…...

目标检测 TaskAlignedAssigner 原理
文章目录 TaskAlignedAssigner 原理和代码使用示例 TaskAlignedAssigner 原理和代码 原理主要是结合预测的分类分数和边界框与真实标注的信息,找出与真实目标最匹配的锚点,为这些锚点分配对应的目标标签、边界框和分数。 TaskAlignedAssigner 是目标检…...
Qt popup窗口半透明背景
半透明弹窗需要paintEvent()接口支持 方法一:使用setStyleSheet设置半透明样式,如果是子窗口,则可注释构建函数内属性设置 class TranslucentWidget : public QWidget { public: explicit TranslucentWidget(QWidget *parent nullptr)…...

游戏:元梦之星游戏开发代码(谢苏)
《元梦之星》是一款轻松社交派对游戏,玩家们可以化身星宝,体验纯粹的游玩乐趣,收获简单的快乐。无论i人e人,都能轻松找到属于自己的社交方式。 《元梦之星》的快乐,可以是闯关夺冠时的激动,谁是狼人推理的巧妙,峡谷3V3打赢团战的爽感。也可以是星梦广场开…...

TCP协议原理与Java编程实战:从连接建立到断开的完整解析
1.TCP协议核心:面向连接的可靠通信基石 TCP(Transmission Control Protocol,传输控制协议)是互联网的“可靠信使”,属于传输层协议,其核心在于面向连接和可靠传输。它通过严谨的握手机制与数据控制逻辑&am…...
Linux的top命令使用
Linux系统中top命令详解及使用技巧 一、基础功能 top命令用于实时监控系统性能和进程活动,可查看以下信息: - CPU使用率 - 内存使用情况 - 进程状态信息 - 系统负载数据 二、使用步骤 1. 打开终端输入命令:top 2. 查看实时更新的数据界面&a…...
Spring Cloud Gateway 限流实践:基于 Redis 令牌桶算法的网关层流量治理
一、引言 在微服务架构中,API 网关作为流量枢纽,需对进入系统的请求进行精细化限流,以保护下游服务免受流量冲击。Spring Cloud Gateway 结合 Redis 实现的令牌桶算法,为网关层限流提供了高效、分布式的解决方案。本文将深入解析其原理、配置及实践优化。 二、技术栈与原…...
可视化大屏实现全屏或非全屏
通过点击按钮实现全屏和非全屏效果展示 代码如下: <template> //点击icon图片进入全屏或非全屏<img :src"screenStatus ? /src/assets/noFull.png : /src/assets/full.png" alt"" click"enterFullScreen" /> </te…...
java8函数式接口(函数式接口的匿名实现类作为某些方法的入参)
文章目录 前置介绍通过 lambda 表达式,使用匿名类,实现函数式接口函数式接口和回调函数的关系函数式接口的应用 前置介绍 是 Java 8 引入的核心概念之一,指的是 仅包含一个抽象方法的接口。它可以被 FunctionalInterface 注解标记࿰…...
linux自有服务
文章目录 [TOC](文章目录)linux自有服务概述systemctl管理服务命令CentOS 7 之前CentOS 7 常用自有服务ntpd或systemd-timesyncd时间同步服务ntp同步服务器原理ntpd时间同步操作systemd-timesyncd同步原理systemd-timesyncd时间同步操作 firewalld防火墙计划任务crontab CentOS…...
UniApp网页版集成海康视频播放器
注意:本人全部集成好后使用最新的海康平台下载插件进行替换后就不能预览视频 使用Uni插件进行集成:海康视频H5播放器组件 - DCloud 插件市场 CSDN资源下载:https://download.csdn.net/download/wangdaoyin2010/90910975 注意:初…...
Filter和Interceptor详解(一文了解执行阶段及其流程)
Filter和Interceptor的区别 Filter(过滤器)和 Interceptor(拦截器)都是用于在请求处理前后插入额外逻辑的组件,下面依次介绍,并额外介绍Spring Gateway的过滤器(GlobalFilter/GatewayFilter&am…...

鸿蒙仓颉开发语言实战教程:实现商城应用详情页
昨天有朋友提到鸿蒙既然有了ArkTs开发语言,为什么还需要仓颉开发语言。其实这个不难理解,安卓有Java和Kotlin,iOS先后推出了Objective-C和Swift,鸿蒙有两种开发语言也就不奇怪了。而且仓颉是比ArkTs更加灵活的语言,虽然…...

GitAny - 無需登入的 GitHub 最新倉庫檢索工具
地址:https://github.com/MartinxMax/gitany GitAny - 無需登入的 GitHub 專案搜尋工具 GitAny 是一款基於 Python 的工具,允許你在無需登入的情況下搜尋當天最新的 GitHub 專案。它支援模糊搜尋、條件篩選以及倉庫資料的視覺化分析。 安裝依賴 $ pip…...

在飞牛nas系统上部署gitlab
在飞牛nas系统上部署gitlab需要使用docker进行部署,如下将介绍详细的部署流程。 文章目录 1. docker镜像2. 拉取镜像3. 运行容器4. 运行和访问gitlab5. 一些小配置5.1 url问题5.2 ssh端口5.3 其他配置 1. docker镜像 首先需要找一个gitlab的docker镜像地址&#x…...

深入理解 Redis 哨兵模式
Redis 哨兵模式深度解析:从原理到实践的全流程指南 在分布式系统架构中,Redis 作为高性能的内存数据库,其哨兵模式(Sentinel)是保障服务高可用性的核心方案。本文将从基础概念、运行机制出发,结合具体配置…...
SQL进阶之旅 Day 4:子查询与临时表优化
文章标题 【SQL进阶之旅 Day 4】子查询与临时表优化 文章内容 开篇:SQL进阶之旅的第4天 在“SQL进阶之旅”系列中,第4天的主题是子查询与临时表优化。这是SQL开发中不可或缺的一部分,尤其在处理复杂查询时,合理使用子查询和临…...

[特殊字符]《Qt实战:基于QCustomPlot的装药燃面动态曲线绘制(附右键菜单/样式美化/完整源码)》
1、将qcustomplot.cpp qcustomplot.h放入工程目录下引入qcustomplot 2、代码 .h #if defined(_MSC_VER) #pragma execution_character_set(...

力扣-最大连续一的个数
1.题目描述 2.题目链接 1004. 最大连续1的个数 III - 力扣(LeetCode) 3.代码解答 class Solution {public int longestOnes(int[] nums, int k) {int zero0,length0;for(int left0,right0;right<nums.length;right){if(nums[right]0){zero;}while…...

无人机避障——深蓝学院浙大栅格地图以及ESDF地图内容
Occupancy Grid Map & Euclidean Signed Distance Field: 【注意】:目的是为了将有噪声的传感器收集起来,用于实时的建图。 Occupancy Grid Map: 概率栅格: 【注意】:由于传感器带有噪声,在实际中基于…...

Postman基础操作
1.Postman是什么? Postman是接口测试的工具,简单来说它能模拟浏览器对服务器的某个接口发起请求并接收响应数据。 1.1 Postman工作原理 2.Postman发送请求 2.1 发送GET请求 我们知道GET请求是没用请求体的,所以我们需要将请求参数写在Param…...

【MPC控制 - 从ACC到自动驾驶】3 MPC控制器设计原理与参数配置:打造ACC的“最强大脑”
【MPC控制 - 从ACC到自动驾驶】MPC控制器设计原理与参数配置:打造ACC的“最强大脑” 在Day 1,我们认识了ACC自适应巡航和MPC这位“深谋远虑的棋手”。Day 2,我们一起给汽车“画像”,建立了它的纵向动力学模型,并把它翻…...

Unity3D仿星露谷物语开发52之菜单页面
1、目标 创建菜单页面,可通过Esc键开启或关闭。 当把鼠标悬停在上面时它会高亮,然后当点击按钮时标签页会被选择。 2、 创建PauseMenuCanvas (1)创建Canvas 在Hierarchy -> PersistentScene -> UI下创建新的Cavans命名为…...
待定事项之存储数据
#### 部署云服务器  ### 部署云服务器完整步骤 1. **连接到云服务器** bash ssh root<服务器IP> 2. **创建项目目录结构** bash mkdir -p /var/www/three/study/待办事项 3. **克隆项目仓库** bash cd /var/www…...
电脑装的数据越多,会不会越重
在这个数字化飞速发展的时代,有一个看似荒诞却又引人深思的问题:电脑装的数据越多,会不会越重? 先来说说大家的普遍认知,我们通常认为数据只是一些虚拟的代码和信息,存放在电脑的硬盘或其他存储设备中&…...