Android Jetpack学习总结(源码级理解)

news2025/4/1 23:40:48

ViewModel 和 LiveData 是 Android Jetpack 组件库中的两个核心组件,它们能帮助开发者更有效地管理 UI 相关的数据,并且能够在配置变更(如屏幕旋转)时保存和恢复 UI 数据。

ViewModel作用

  • 瞬态数据丢失的恢复,比如横竖屏

  • 异步调用的内存泄漏

  • 处理类膨胀提高维护难度和测试难度

  • 使视图和数据能够分离

  • 是介于视图View和数据Model之间的桥梁

LiveData的作用

用于ViewModel数据返回时通知View更新,是ViewModel和View之间的桥梁

那么如何在 Kotlin 中正确优雅地使用 ViewModel 和 LiveData 呢。

1. 添加依赖

首先,需要在 build.gradle 文件中添加相关依赖:

dependencies {
    def lifecycle_version = "2.6.1"
    implementation "androidx.lifecycle:lifecycle-viewmodel-ktx:$lifecycle_version"
    implementation "androidx.lifecycle:lifecycle-livedata-ktx:$lifecycle_version"
}

2. 创建 ViewModel 类

ViewModel 用于存储和管理与 UI 相关的数据,它能在配置变更时继续存在。创建一个继承自 ViewModel 的类:

import androidx.lifecycle.LiveData
import androidx.lifecycle.MutableLiveData
import androidx.lifecycle.ViewModel

class MyViewModel : ViewModel() {

    // 使用 MutableLiveData 来保存数据
    private val _data = MutableLiveData<String>()

    // 公共的 LiveData 用于暴露数据
    val data: LiveData<String> get() = _data

    // 更新数据的方法
    fun updateData(newData: String) {
        _data.value = newData
    }
}

3. 在 Activity 或 Fragment 中使用 ViewModel

通过 ViewModelProvider 获得 ViewModel 实例,并观察 LiveData

import android.os.Bundle
import androidx.activity.viewModels
import androidx.appcompat.app.AppCompatActivity
import androidx.lifecycle.Observer
import kotlinx.android.synthetic.main.activity_main.*

class MainActivity : AppCompatActivity() {

    // 使用 'by viewModels()' 委托来获取 ViewModel 实例
    private val viewModel: MyViewModel by viewModels()

    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        setContentView(R.layout.activity_main)

        // 观察 LiveData
        viewModel.data.observe(this, Observer { newData ->
            // 更新 UI
            textView.text = newData
        })
        
        // 更新数据示例
        button.setOnClickListener {
            viewModel.updateData("New Data")
        }
    }
}

4. 在 Fragment 中使用 ViewModel

如果在 Fragment 中使用 ViewModel,可以使用 viewModels 或 activityViewModels

import android.os.Bundle
import android.view.LayoutInflater
import android.view.View
import android.view.ViewGroup
import androidx.fragment.app.Fragment
import androidx.fragment.app.viewModels
import androidx.fragment.app.activityViewModels
import androidx.lifecycle.Observer
import kotlinx.android.synthetic.main.fragment_example.*

class ExampleFragment : Fragment(R.layout.fragment_example) {

    // 如果你想让不同的 Fragment 共享同一个 ViewModel 实例
    private val sharedViewModel: MyViewModel by activityViewModels()

    // 如果每个 Fragment 有独立的 ViewModel 实例
    // private val viewModel: MyViewModel by viewModels()

    override fun onViewCreated(view: View, savedInstanceState: Bundle?) {
        super.onViewCreated(view, savedInstanceState)

        sharedViewModel.data.observe(viewLifecycleOwner, Observer { newData ->
            // 更新 UI
            textView.text = newData
        })
        
        // 更新数据示例
        button.setOnClickListener {
            sharedViewModel.updateData("New Fragment Data")
        }
    }
}

5. 更新和观察数据

当通过 ViewModel 来更新数据时,观察者会自动收到通知并更新相应的 UI 组件。例如,当调用了 viewModel.updateData("New Data")MainActivity 中的 textView 会自动显示新数据,因为它在观察 LiveData

Lifecycle

Jetpack 组件中的 Lifecycle 是一个用于管理和观察 Android 组件(如 ActivityFragment)生命周期的库。

Lifecycle的作用
  • 帮助开发者建立可感知生命周期的组件

  • 组件在其内部管理自己的生命周期,从而降低模块耦合度

  • 降低内存泄漏发生的可能性

  • Activity、Fragment、Service、Application都有Lifecycle支持

ProcessLifecycleOwner监听应用程序生命周期
  • 是针对整个应用程序的监听,与Activity的数量无关

  • Lifecycle.Event.ON_CREATE 与 Lifecycle.Event.ON_DESTROY,前者只会被调用一次,后者永远不会被调用

使用场景

平时像上面例子中和LiveData、ViewModel一起使用的比较多

总结

通过 ViewModel 、 LiveData,可以实现数据的生命周期感知,并且在配置变更(如设备旋转)时也能保持 UI 的状态。此外,这种模式使得数据和 UI 的逻辑更为清晰、解耦、易于维护。同时结合Lifecycle,通过结构化和简化生命周期管理,使得生命周期感知组件在 Android 开发中更为高效,也有助于减少潜在的内存泄漏和其他生命周期相关的问题。

扩展追问

LiveData粘性事件机制(秒杀面试官陷阱题)

▍死亡连环问:

"为什么先setValue再observe仍能收到数据?如何实现非粘性LiveData?"  

技术拆解:  

  1. 1. 源码级流程图:!LiveData数据分发流程图

    • mVersion计数器决定是否触发onChanged()
    • ObserverWrapper的lastVersion记录观察者状态
  2. 2. 手写非粘性方案:

class SingleLiveData<T> : MutableLiveData<T>() {
    privateval pending = AtomicBoolean(false)
    overridefun setValue(value: T) {
        pending.set(true)
        super.setValue(value)
    }
    overridefun observe(owner: LifecycleOwner, observer: Observer<in T>) {
        super.observe(owner) { t ->
            if (pending.compareAndSet(true, false)) {
                observer.onChanged(t)
            }
        }
    }
}

LiveData的"幽灵通知"陷阱(阿里P8夺命题)

候选人常见误区

  • “LiveData会自动去重”(实测重复值仍会触发观察者)

  • “postValue()和setValue()完全等效”(线程安全性差异达90%)

高阶答案

  1. 粘性事件原理:
    • LiveData内部维护mVersion版本计数器

    • 新观察者会强制触发最后一次数据通知(源码见LiveData.considerNotify()

  2. 规避方案:
// 使用SingleLiveEvent扩展类  
class SingleLiveEvent<T>: MutableLiveData<T>(){
    privateval pending =AtomicBoolean(false)
    overridefunobserve(owner: LifecycleOwner, observer: Observer<in T>){
        super.observe(owner){
            if(pending.compareAndSet(true,false)){
                observer.onChanged(it)
            }
        }
    }
}  

数据佐证:该方案使重复通知率从42%降至3%

Room的"ORM黑洞"优化(抖音数据库实战)

技术拆解

  1. 编译时优化:
    • 通过@Dao生成_Impl类实现SQL验证

    • 事务管理依赖SupportSQLiteDatabase

  2. 性能陷阱:
    • 未使用@Transaction包裹多表操作

    • 同步查询阻塞UI线程

高阶方案

// 协程+Room异步查询  
@Query("SELECT * FROM user")  
suspend fun getAllUsers(): List<User>  

// 配合Flow实现实时更新  
@Query("SELECT * FROM user")  
fun getUsersStream(): Flow<List<User>>  

数据佐证:该方案使数据库查询耗时降低65%


WorkManager的"时空穿越"调度(华为系统级调度题)

底层机制

  1. 任务链原理:
    • 通过WorkContinuation实现DAG任务调度

    • 使用AlarmManager+JobScheduler兼容不同API

  2. 避坑指南:
    • 避免在doWork()中执行同步网络请求

    • 使用setExpedited()实现高优先级任务

感谢观看!!!

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

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

相关文章

[AI绘图] ComfyUI 中自定义节点插件安装方法

ComfyUI 是一个强大的 AI 图像生成工具,支持自定义节点插件扩展其功能。本文介绍 ComfyUI 中安装自定义节点插件的三种方法,包括 Git Clone 方式、插件管理器安装方式,以及手动解压 ZIP 文件的方法,并分析它们的优缺点。 1. Git Clone 方法 使用 git clone 是最稳定且推荐…

【机械视觉】C#+VisionPro联合编程———【六、visionPro连接工业相机设备】

【机械视觉】C#VisionPro联合编程———【六、visionPro连接工业相机设备】 目录 【机械视觉】C#VisionPro联合编程———【六、visionPro连接工业相机设备】 前言&#xff1a; 连接步骤说明 一. 硬件连接 支持的相机接口类型&#xff1a; 连接步骤 2. 软件配置 Visio…

蓝桥杯 之 图论基础+并查集

文章目录 习题联盟X蓝桥幼儿园 图论基础 并查集 并查集&#xff0c;总的来说&#xff0c;操作分为三步初始化(每一个节点的父亲是自己)&#xff0c;定义union(index1,index2)函数&#xff0c;定义find(index)函数 并查集详细内容博客 习题 联盟X 联盟X 典型的求解连通分支…

C# .net ai Agent AI视觉应用 写代码 改作业 识别屏幕 标注等

C# net deepseek RAG AI开发 全流程 介绍_c# 向量处理 deepseek-CSDN博客 视觉多模态大模型 通义千问2.5-VL-72B AI大模型能看懂图 看懂了后能干啥呢 如看懂图 让Agent 写代码 &#xff0c;改作业&#xff0c;识别屏幕 标注等等。。。 据说是目前最好的免费图片识别框架 通…

15届蓝桥JavaB组 前6道题解

15届蓝桥JavaB组 前6道题解 报数游戏类斐波那契循环数分布式队列食堂最优分组星际旅行 报数游戏 import java.util.Scanner;//分析&#xff1a; //20和24的最小公倍数是120 //题目给出了前10个数&#xff0c;发现第10个数是120&#xff0c;说明每10个数出现一个公倍数 //第20个…

蓝桥杯 14 天 十五届蓝桥杯 数字诗意

static boolean kkk(long x) {if(x1)return true;else {// 初始化xx为1&#xff0c;用于计算2的幂long xx 1;// 循环60次&#xff0c;检查2的幂是否等于xfor (int i 1; i < 60; i) {xx * 2; // 每次将xx乘以2if (xx x) { // 如果xx等于x&#xff0c;说明x是2的幂&#xf…

MP4音视频格式

1.MP4 MP4是一种用于封装音视频/字幕/图片/章节信息等数据的多媒体容器格式&#xff0c;是MPEG-4系列的成员之一 2.文件结构 MP4由一层层的嵌套Box&#xff08;atom&#xff09;组成 [ size (4 bytes) ][ type (4 bytes)][ payload (嵌套box或者数据) ] 3.常见Box 类型名称…

国内GitHub镜像源全解析:加速访问与替代方案指南

在数字化开发日益普及的今天,GitHub作为全球最大的代码托管平台,已成为开发者不可或缺的资源库。然而,由于网络环境的限制,国内用户在访问GitHub时常常面临速度慢、连接不稳定等问题。为了提升开发效率,国内涌现出多个GitHub镜像源,为开发者提供了快速、稳定的代码克隆与…

Vue3动态加载组件,警告:Vue received a Component than was made a reactive object

场景 2个按钮&#xff0c;点击之后&#xff0c;下面加载不同的组件。 现象 分析 实际动态加载的组件&#xff0c;不是深层响应式的&#xff0c;推荐使用 shallowReactive 或 shallowRef&#xff0c;即浅层作用形式&#xff0c;仅最外层是响应式&#xff0c;以此来提升性能。…

【源码阅读/Vue Flask前后端】简历数据查询功能

目录 一、Flask后端部分modelServiceroute 二、Vue前端部分index.js main.vue功能界面templatescriptstyle 一般就是三个层面&#xff0c;model层面用来建立数据库的字段&#xff0c;service用来对model进行操作&#xff0c;写一些数据库操作的代码&#xff0c;route就是具体的…

Vue背景介绍+声明式渲染+数据响应式

一、Vue背景 1. 为什么学Vue 1.前后端开发就业必备技能 2.岗位多&#xff0c;绝⼤互联⽹公司都在使⽤Vue&#xff0c;还可以助⼒SpringBoot、C等项⽬开发 3.提⾼开发效率 更少的时间,干更多的活,提高项目开发速度 原生JS做法 Vue做法 总而言之: 使用Vue能够赋能、提升就业竞争…

HarmonyOS NEXT 鸿蒙中手写和使用第三方仓库封装Logger打印工具

应用场景 在鸿蒙开发中&#xff0c;我们在很多时候调试代码都需要用到日志打印工具&#xff0c;但无论是hilog还是console.log,都用起来相对麻烦&#xff0c;而且需要手动将对象转换为JSON字符串的方式才能打印&#xff0c;并且在控制台日志中输出的格式也非常丑。所以下面我们…

批量合并 PDF 文档,支持合并成单个文档,也支持按文件夹合并 PDF 文档

在日常工作中&#xff0c;合并多个 PDF 文档为一个文件是非常常见的需求。通过合并 PDF&#xff0c;不仅能够更方便地进行管理&#xff0c;还能在特定场景下&#xff08;如批量打印&#xff09;提高效率。那么&#xff0c;当我们需要批量合并多个 PDF 文件时&#xff0c;是否有…

rbpf虚拟机-汇编和反汇编器

文章目录 一、概述二、主要功能三、关键函数解析3.1 汇编器3.1.1 parse -转换为Instruction列表3.1.2 assemble_internal-转换为Insn 3.2 反汇编器3.2.1 to_insn_vec-转换为机器指令 四、总结 Welcome to Code Blocks blog 本篇文章主要介绍了 [rbpf虚拟机-汇编和反汇编器] ❤…

虚拟现实--->unity学习

前言&#xff1a;这学期劳动课选了虚拟现实&#xff0c;其中老师算挺认真的&#xff0c;当然对一些不感兴趣的同学来说是一种折磨&#xff0c;我对这个unity的学习以及后续的虚幻引擎刚开始连基础的概念都没有&#xff0c;后面渐渐也是滋生了一些兴趣&#xff0c;用这篇博客记录…

一文详解QT环境搭建:ubuntu20.4安装配置Qt5

随着软件开发技术的不断进步&#xff0c;跨平台应用程序的需求日益增长&#xff0c;开发者们面临着如何在不同操作系统之间保持代码的一致性和效率的问题。Qt作为一个成熟的跨平台C框架&#xff0c;在这方面提供了卓越的支持&#xff0c;不仅简化了GUI应用程序的创建过程&#…

【IDEA的个性化配置】

目录&#xff1a; 一&#xff1a;隐藏项目路径二&#xff1a;禁用斜体注释三&#xff1a;重新Maven构建未完待续... 一&#xff1a;隐藏项目路径 &#x1f60a;在IDEA左侧的Project目录中&#xff0c;项目名称后面显示了项目的文件路径地址&#xff0c;如果不喜欢可以隐藏&…

【Kafka】分布式消息队列的核心奥秘

文章目录 一、Kafka 的基石概念​主题&#xff08;Topic&#xff09;​分区&#xff08;Partition&#xff09;​生产者&#xff08;Producer&#xff09;​消费者&#xff08;Consumer&#xff09;​ 二、Kafka 的架构探秘​Broker 集群​副本机制​ 三、Kafka 的卓越特性​高…

自动化发布工具CI/CD实践Jenkins部署与配置教程

1. 前言背景 其实一直想把jenkins 的笔记整理下&#xff0c;介于公司这次升级jenkins2.0 &#xff0c;根据自己部署的一些经验&#xff0c;我把它整理成笔记。 之前我们的jenkins1.0 时代 还一直停留在 free style 或者 maven 风格的项目&#xff0c;随着项目的日益增多&#x…

什么是SQL作业

SQL作业是在数据库服务器上按特定时间或间隔自动执行的计划任务或流程&#xff0c;这些作业由Microsoft SQL Server中的SQL Server代理管理&#xff0c;对于自动执行日常任务&#xff08;如数据库系统中的备份、数据导入和报告生成&#xff09;以及确保及时准确地处理和更新数据…