Livedata源码详细解析-面试这么讲就ok

简介: Livedata源码详细解析-面试这么讲就ok

时序图


不太擅长画图,勉强画了一幅

如果您对Livedata的源码有一定了解看了这个图应该就差不多了。

如果对Livedata源码了解不多,那么粗略看一下图,然后直接看后面的逻辑。看完逻辑回过头再细看时序图效果更佳

image.png


Livedata 方法调用简单说明


既然是源码解析,自然入口就是我们的方法调用,这里简单把注册 Livedata 的数据监听和 Livedata.setValue 方法进行简单说明


注册监听代码

mapViewModel.mapLiveData.observe(this,{//这个回调中会收到数据更改的监听
            logEE(it)
            tv_map.text=it
        })
复制代码


数据更改,发送数据

val userLivedata =MutableLiveData<User>()
 fun sendData() {
        userLivedata.value=User("李白",1200)//这句代码发送最新数据
    }
复制代码

下一节就是 Livedata 的工作流程简单分析了


Livedata 代码逻辑


我主要把代码逻辑分为了三部分:

  1. 业务代码 Livedata.observe 注册数据观察
  2. setValue 发送最新数据
  3. 用户视图回到前台自动进行数据分发


第一部分逻辑:

基本的逻辑就是我们注册数据监听的时候,会传入一个 LifeCycle 的数据监听者接口。Livedata 的 observe 方法会将接口封装成一个 LifecycleBoundObserver对象,并且注册 LifeCycle 的生命周期监听。


第二部分逻辑:

当我们调用 Livedata.setValue 方法的时候,会逐个遍历每一个注册的 LifecycleBoundObserver,然后分别调用 Observer 的 onChange 回调方法向我们的业务数据监听返回数据


第三部分逻辑:

因为我们注册了 LifeCycle 的生命周期,当用户视图回到前台的时候会回调 LifecycleBoundObserver 的 onStateChanged 方法。onStateChanged 方法内会进行数据的分发


本节讲了 Livedata 的基本逻辑,下一节主角就是代码了


结合代码讲解逻辑


注册 Livedata 监听

  1. observe 方法体
public void observe(@NonNull LifecycleOwner owner, @NonNull Observer<? super T> observer) {
        assertMainThread("observe");
        if (owner.getLifecycle().getCurrentState() == DESTROYED) {//如果生命周期已经被销毁,return。这说明了使用Livedata不会发生内存泄露的原因
            // ignore
            return;
        }
        LifecycleBoundObserver wrapper = new LifecycleBoundObserver(owner, observer);//这个对象是LifecycleEventObserver的一个实现,是我们注册生命周期组件的观察者
        ObserverWrapper existing = mObservers.putIfAbsent(observer, wrapper);//将wrapper加入到map中,得到上一个wrapper,
        if (existing != null && !existing.isAttachedTo(owner)) {//isAttachedTo:上一个wrapper关联的生命周期如果不是当前生命周期,就抛出异常
            throw new IllegalArgumentException("Cannot add the same observer"
                    + " with different lifecycles");
        }
        if (existing != null) {如果上一个wrapper不为空就return(因为会复用上一个wrapper)
            return;
        }
        owner.getLifecycle().addObserver(wrapper);//如果上一个为空则说明是第一次,需要注册生命周期监听
    }
复制代码


Livedata.setValue

  1. 首先看一下 setValue 方法体:
protected void setValue(T value) {
        assertMainThread("setValue");
        mVersion++;//数据的版本,每次调用setValue都会++
        mData = value;
        dispatchingValue(null);//分发数据
    }
复制代码


  1. 我们发现发法最后调用了 dispatchingValue 方法,那么就再看一下 dispatchingValue 方法体吧
void dispatchingValue(@Nullable ObserverWrapper initiator) {
          if (mDispatchingValue) {// 1、如果正在分发中则返回
              mDispatchInvalidated = true;// 2、目的是如果有新的setValue方法被调用,会在步骤4中跳出for循环,重新再进行一次分发
              return;
          }
          mDispatchingValue = true;、// 3、设置正在分发中状态为true,防止1中重复进入分发消息
          do {
              mDispatchInvalidated = false;// 4、进入循环后设置mDispatchInvalidated为false,如果用户重新setValue需要将mDispatchInvalidated重新设置为true,一边可以在上一轮分发结束后继续走一轮分发
              if (initiator != null) {
                  considerNotify(initiator);// 5、分发消息
                  initiator = null;
              } else {
                  for (Iterator<Map.Entry<Observer<? super T>, ObserverWrapper>> iterator =
                          mObservers.iteratorWithAdditions(); iterator.hasNext(); ) {
                      considerNotify(iterator.next().getValue());// 6、循环中分发消息
                      if (mDispatchInvalidated) {//6、如果这里为true说明用户又设置了一次value,那么跳出for循环,重新执行外层的dowhile循环
                          break;
                      }
                  }
              }
          } while (mDispatchInvalidated);
          mDispatchingValue = false;//7、消息分发完成,设置正在分发中状态为false
      }
复制代码


3.通过分析,我们发现 dispatchingValue 方法内最终再次分发消息的方法是 considerNotify,方法体:

private void considerNotify(ObserverWrapper observer) {
              if (!observer.mActive) {//1、如果当前lifecycle不是活跃状态就返回不进行消息分发,销毁会在生命周期最近变活跃的时候自动分发
                  return;
              }
              if (!observer.shouldBeActive()) {//2、如果lifecycle状态为started||resumed
                  observer.activeStateChanged(false);//3、激活状态,activeStateChanged方法里面会再次调用dispatchValue分发消息,一般用作livedata从不活跃状态回到活跃状态的时候自动分发消息
                  return;
              }
              if (observer.mLastVersion >= mVersion) {//4、如果true,说明第6步中有更小版本的消息正在调用,所以返回
                  return;
              }
              observer.mLastVersion = mVersion;//5、记住当前正在分发value的版本,用于第四部中的逻辑判断
              observer.mObserver.onChanged((T) mData);//6、调用注册livedata的observer的onChanged通知用户更新数据
          }
复制代码


considerNotify 方法中调用了 mObserver.onChanged 后就会回调我们业务代码中的消息注册监听,整个数据流转的逻辑就完成了


视图重新回到前台

逻辑同 Livedata.setValue

我们知道我们是通过 LifeCycle 生命周期组件来监听生命周期的。

监听生命周期的接口实现就是 LifecycleBoundObserver 类,当生命周期发生改变的时候会回调 LifecycleBoundObserver 类的方法 onStateChanged:

  1. onStateChanged 方法体如下
public void onStateChanged(@NonNull LifecycleOwner source,
                @NonNull Lifecycle.Event event) {
            Lifecycle.State currentState = mOwner.getLifecycle().getCurrentState();
            if (currentState == DESTROYED) {//如果生命周期已经销毁,那么移除监听者
                removeObserver(mObserver);
                return;
            }
            Lifecycle.State prevState = null;
            while (prevState != currentState) {//回调的生命周期状态和当前不同,则进行数据回调
                prevState = currentState;
                activeStateChanged(shouldBeActive());//这个方法是关键,他内部会进行数据的分发
                currentState = mOwner.getLifecycle().getCurrentState();//重置生命周期状态
            }
        }
复制代码


  1. activeStateChanged 方法体
void activeStateChanged(boolean newActive) {
            if (newActive == mActive) {
                return;
            }
            mActive = newActive;
            changeActiveCounter(mActive ? 1 : -1);
            if (mActive) {
                dispatchingValue(this);//看到这里你直接去找上一节Livedata.setValue的源码分析就ok了
            }
        }
复制代码


postValue

实际没啥好说的,就是将线程切回到主线程,然后调用 Livedata.setValue 更新数据而已。



相关文章
|
2月前
|
存储 缓存 NoSQL
Redis常见面试题全解析
Redis面试高频考点全解析:从过期删除、内存淘汰策略,到缓存雪崩、击穿、穿透及BigKey问题,深入原理与实战解决方案,助你轻松应对技术挑战,提升系统性能与稳定性。(238字)
|
4月前
|
存储 安全 测试技术
Python面试题精选及解析
本文详解Python面试中的六大道经典问题,涵盖列表与元组区别、深浅拷贝、`__new__`与`__init__`、GIL影响、协程原理及可变与不可变类型,助你提升逻辑思维与问题解决能力,全面备战Python技术面试。
181 0
|
2月前
|
监控 Java 关系型数据库
面试性能测试总被刷?学员真实遇到的高频问题全解析!
面试常被性能测试题难住?其实考的不是工具,而是分析思维。从脚本编写到瓶颈定位,企业更看重系统理解与实战能力。本文拆解高频面试题,揭示背后考察逻辑,并通过真实项目训练,帮你构建性能测试完整知识体系,实现从“会操作”到“能解决问题”的跨越。
|
6月前
|
存储 安全 Java
2025 最新史上最全 Java 面试题独家整理带详细答案及解析
本文从Java基础、面向对象、多线程与并发等方面详细解析常见面试题及答案,并结合实际应用帮助理解。内容涵盖基本数据类型、自动装箱拆箱、String类区别,面向对象三大特性(封装、继承、多态),线程创建与安全问题解决方法,以及集合框架如ArrayList与LinkedList的对比和HashMap工作原理。适合准备面试或深入学习Java的开发者参考。附代码获取链接:[点此下载](https://pan.quark.cn/s/14fcf913bae6)。
3587 48
|
6月前
|
缓存 NoSQL Java
Java Redis 面试题集锦 常见高频面试题目及解析
本文总结了Redis在Java中的核心面试题,包括数据类型操作、单线程高性能原理、键过期策略及分布式锁实现等关键内容。通过Jedis代码示例展示了String、List等数据类型的操作方法,讲解了惰性删除和定期删除相结合的过期策略,并提供了Spring Boot配置Redis过期时间的方案。文章还探讨了缓存穿透、雪崩等问题解决方案,以及基于Redis的分布式锁实现,帮助开发者全面掌握Redis在Java应用中的实践要点。
371 6
|
6月前
|
NoSQL Java 微服务
2025 年最新 Java 面试从基础到微服务实战指南全解析
《Java面试实战指南:高并发与微服务架构解析》 本文针对Java开发者提供2025版面试技术要点,涵盖高并发电商系统设计、微服务架构实现及性能优化方案。核心内容包括:1)基于Spring Cloud和云原生技术的系统架构设计;2)JWT认证、Seata分布式事务等核心模块代码实现;3)数据库查询优化与高并发处理方案,响应时间从500ms优化至80ms;4)微服务调用可靠性保障方案。文章通过实战案例展现Java最新技术栈(Java 17/Spring Boot 3.2)的应用.
541 9
|
6月前
|
缓存 算法 NoSQL
校招 Java 面试高频常见知识点深度解析与实战案例详细分享
《2025校招Java面试核心指南》总结了Java技术栈的最新考点,涵盖基础语法、并发编程和云原生技术三大维度: 现代Java特性:重点解析Java 17密封类、Record类型及响应式Stream API,通过电商案例演示函数式数据处理 并发革命:对比传统线程池与Java 21虚拟线程,详解Reactor模式在秒杀系统中的应用及背压机制 云原生实践:提供Spring Boot容器化部署方案,分析Spring WebFlux响应式编程和Redis Cluster缓存策略。
170 0
|
6月前
|
存储 缓存 安全
Java 集合容器常见面试题及详细解析
本文全面解析Java集合框架,涵盖基础概念、常见接口与类的特点及区别、底层数据结构、线程安全等内容。通过实例讲解List(如ArrayList、LinkedList)、Set(如HashSet、TreeSet)、Map(如HashMap、TreeMap)等核心组件,帮助读者深入理解集合容器的使用场景与性能优化。适合准备面试或提升开发技能的开发者阅读。
120 0
|
6月前
|
存储 Java 数据库
应届生面试高频 Java 基础问题及详细答案解析
摘要: Java数据类型分为基本类型(如int、float等)和引用类型(如类、数组)。final可修饰类、方法和变量,使其不可继承、重写或修改。static用于类级别的变量和方法,共享于所有实例。&quot;==&quot;比较基本类型的值或引用类型的地址,而equals比较对象内容(需重写)。Java只有值传递,对于引用类型传递的是地址副本。String对象不可变,拼接操作会创建新对象而非修改原对象。Java 10的var支持类型推断,Java 16的Record提供不可变类简化。
153 0
|
6月前
|
存储 安全 Java
应届生面试高频 Java 基础问题及实操示例解析
本文总结了Java基础面试中的高频考点,包括数据类型分类、final修饰符的三种用途、static关键字特性、==与equals的区别、Java只有值传递的特性、String的不可变性、Error与Exception的差异、程序初始化顺序规则,以及IO流的字节流/字符流分类。每个问题都配有简明定义和典型示例,如用final修饰变量示例、static方法调用限制说明等,帮助应聘者快速掌握核心概念和实际应用场景。
135 0

热门文章

最新文章

推荐镜像

更多
  • DNS