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

相关推荐
执风挽^1 天前
Python基础编程题2
开发语言·python·算法·visual studio code
程序员泠零澪回家种桔子1 天前
Spring AI框架全方位详解
java·人工智能·后端·spring·ai·架构
Z9fish1 天前
sse哈工大C语言编程练习20
c语言·开发语言·算法
CodeCaptain1 天前
nacos-2.3.2-OEM与nacos3.1.x的差异分析
java·经验分享·nacos·springcloud
萧鼎1 天前
Python 包管理的“超音速”革命:全面上手 uv 工具链
开发语言·python·uv
Anastasiozzzz1 天前
Java Lambda 揭秘:从匿名内部类到底层原理的深度解析
java·开发语言
骇客野人1 天前
通过脚本推送Docker镜像
java·docker·容器
刘琦沛在进步1 天前
【C / C++】引用和函数重载的介绍
c语言·开发语言·c++
机器视觉的发动机1 天前
AI算力中心的能耗挑战与未来破局之路
开发语言·人工智能·自动化·视觉检测·机器视觉
铁蛋AI编程实战1 天前
通义千问 3.5 Turbo GGUF 量化版本地部署教程:4G 显存即可运行,数据永不泄露
java·人工智能·python