Android 内存泄漏 -> LiveData如何解决ViewMode和Activity/Fragment之间的内存泄漏

LiveData 是如何解决的?(源码分析)

LiveData 没有直接持有 Activity,而是持有了一个 Observer(观察者),而这个 Observer 被包装在一个 生命周期感知器 中。

核心机制: 当 Activity 变为 DESTROYED 状态时,LiveData 会自动把自己从观察者列表中移除。引用链断裂,Activity 就可以被回收了。

observe 方法:绑定的开始,当你调用 viewModel.liveData.observe(this, observer) 时:

java 复制代码
// LiveData.java
@MainThread
public void observe(@NonNull LifecycleOwner owner, @NonNull Observer<? super T> observer) {
    // 1. 如果当前页面已经销毁了,直接忽略,不注册
    if (owner.getLifecycle().getCurrentState() == DESTROYED) {
        return;
    }
    
    // 2. 将 owner (Activity) 和 observer 包装成一个 LifecycleBoundObserver
    LifecycleBoundObserver wrapper = new LifecycleBoundObserver(owner, observer);
    
    // 3. 保存到 map 中
    ObserverWrapper existing = mObservers.putIfAbsent(observer, wrapper);
    
    // 4. 【关键】将这个 wrapper 注册到 Activity 的生命周期中
    // 这样 wrapper 就能收到 onStart, onStop, onDestroy 等回调
    owner.getLifecycle().addObserver(wrapper);
}

LifecycleBoundObserver:核心内部类,实现了 LifecycleEventObserver,它会监听 Activity 的生命周期变化

java 复制代码
// LiveData.java 的内部类
class LifecycleBoundObserver extends ObserverWrapper implements LifecycleEventObserver {
    @NonNull final LifecycleOwner mOwner;

    LifecycleBoundObserver(@NonNull LifecycleOwner owner, Observer<? super T> observer) {
        super(observer);
        mOwner = owner;
    }

    @Override
    public void onStateChanged(@NonNull LifecycleOwner source, @NonNull Lifecycle.Event event) {
        // 【核心代码在这里!】
        // 每次生命周期变化都会回调这里
        
        // 1. 自动断开引用机制
        // 如果当前状态是 DESTROYED (Activity 正在销毁)
        if (mOwner.getLifecycle().getCurrentState() == DESTROYED) {
            // 调用 removeObserver,把自己从 LiveData 的列表中移`在这里插入代码片`除
            removeObserver(mObserver);
            return;
        }
        
        // 2. 只有在活跃状态 (STARTED/RESUMED) 才发送数据更新
        activeStateChanged(shouldBeActive());
    }
}

3. removeObserver:彻底断开

java 复制代码
// LiveData.java
public void removeObserver(@NonNull final Observer<? super T> observer) {
    ObserverWrapper removed = mObservers.remove(observer);
    if (removed == null) {
        return;
    }
    // 取消监听生命周期,彻底解绑
    removed.detachObserver();
    removed.activeStateChanged(false);
}

总结:LiveData 的高明之处

LiveData 并不是简单地"持有引用然后置空",它引入了 Inversion of Control (控制反转)ViewModel 不认识 ActivityViewModel 里的 LiveData 只是一个数据容器,它不知道谁在观察它。Activity 自己管理自己:Activity 通过 observe 告诉 LiveData:"我有生命周期,我死的时候(DESTROYED),请不要再理我,把我的引用扔掉。"

自动清理:
Activity A (Old) 销毁 -> 触发 LifecycleBoundObserver.onStateChanged -> 检测到 DESTROYED -> removeObserver -> 引用断开,Activity A 被回收。
Activity A (New) 重建 -> 调用 observe -> 注册新的 Observer -> ViewModel 连接到新的 Activity

相关推荐
2301_7736436212 小时前
ceph池
开发语言·ceph·python
两年半的个人练习生^_^12 小时前
JMM 进阶:彻底理解 CAS 实现原理
java·开发语言
wuminyu12 小时前
Java锁机制之park和unpark源码剖析
java·linux·c语言·jvm·c++
Refrain_zc12 小时前
Android 英语口语评测:从录音采集到单词级着色反馈的完整技术方案
kotlin
半个烧饼不加肉12 小时前
JS 底层探究-- 事件循环
开发语言·前端·javascript
W_LuYi18513 小时前
手撸极简zkEVM验证器:RISC-V电路实践
java·risc-v
asdfg125896313 小时前
C 语言中产生伪随机数的标准做法
c语言·开发语言
AI人工智能+电脑小能手13 小时前
【大白话说Java面试题 第102题】【并发篇】第2题:volatile 能否保证线程安全?
java·安全·面试
KobeSacre13 小时前
JUC 概述
java·开发语言
小bo波13 小时前
形式化方法 × UML
java·软件工程·uml·面向对象·形式化方法·tla+