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

相关推荐
代码中介商几秒前
C++ STL 容器完全指南(二):vector 深入与 stringstream 实战
开发语言·c++
scott.cgi1 小时前
Unity直接编译Java文件作为插件,导致失败的两个打包设置
java·unity·unity调用java·unity的java文件·unity的android插件·unity调用android·unity加载java代码
澈2075 小时前
C++并查集:高效解决连通性问题
java·c++·算法
郝学胜-神的一滴6 小时前
Qt 入门 01-01:从零基础到商业级客户端实战
开发语言·c++·qt·程序人生·软件构建
测试员周周6 小时前
【Appium 系列】第06节-页面对象实现 — LoginPage 实战
开发语言·前端·人工智能·python·功能测试·appium·测试用例
alexhilton6 小时前
Android上的ZeroMQ:用发布/订阅模式连接Linux服务
android·kotlin·android jetpack
2401_873479407 小时前
运营活动被薅羊毛怎么防?用IP查询+设备指纹联动封堵漏洞
java·网络·tcp/ip·github
ShiJiuD6668889997 小时前
大事件板块一
java
摇滚侠7 小时前
@Autowired 和 @Resource 的区别
java·开发语言
风别鹤7 小时前
Cocos Creator无法识别Android SDK
android