Android 内存泄漏 -> ViewModel持有Activity/Fragment导致的内存泄漏

问题一:ViewModel 持有构造它的 Activity/Fragment,会导致内存泄漏吗?

答案:是的,一定会导致内存泄漏(除非你手动清空,但那很难维护)。为什么?

这依然是 生命周期不对等 造成的。

  • 引用的本质:当你把 Activitythis 传给 ViewModel 时,ViewModel 内部就有了一个指向该 Activity 内存地址的指针(强引用)。
  • 存活时间差异:
    • Activity:是"短命"的。屏幕旋转、系统语言切换、分屏模式、或者系统内存不足时,它都会被销毁(onDestroy)。
    • ViewModel:是"长命"的。它的设计目的就是为了在 Activity 重建期间存活下来。只有当 Activity 彻底退出(按返回键退出或调用 finish())时,ViewModel 才会清除。
  • 场景推演(旋转屏幕):
    Activity A (实例1) 创建了 ViewModel,并将 this (实例1) 传给它。用户旋转屏幕。系统销毁 Activity A (实例1)
  • 关键点来了:系统本该回收 (GC) 实例1 的内存。但是,ViewModel 还活着,并且手里死死抓着 实例1 的引用。
    结果:GC 无法回收 实例1。系统创建了 Activity A (实例2),并复用了同一个 ViewModel。此时,内存里有:正在显示的 实例2 + 无法回收的僵尸 实例1(包含它的 View、图片等)。这就构成了标准的内存泄漏。

源码分析

核心逻辑:ViewModel 被存储在 ViewModelStore 中,而 ViewModelStore 的生命周期 长于 Activity 的配置重建(如旋转)。
Activity 销毁时: 系统调用 onRetainNonConfigurationInstance()。这里会把 mViewModelStore 保存起来,不销毁。

java 复制代码
// ComponentActivity.java
// 1. Activity 销毁时:系统特意把 ViewModelStore 存起来
public final Object onRetainNonConfigurationInstance() {
    // ...
    NonConfigurationInstances nci = new NonConfigurationInstances();
    // 【核心罪魁祸首】:把包含 ViewModel 的 Store 保存到 nci 对象中
    nci.viewModelStore = mViewModelStore; 
    return nci; // 返回给系统进程保存,不随 Activity 销毁
}

Activity 重建时: 在 onCreate 中,系统把之前保存的 ViewModelStore 又还给了新 Activity

kotlin 复制代码
// ComponentActivity.java
// 2. 新 Activity 重建时:把 ViewModelStore 取回来
protected void onCreate(@Nullable Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    NonConfigurationInstances nc = (NonConfigurationInstances) getLastNonConfigurationInstance();
    if (nc != null) {
        // 【恢复】:新 Activity 拿到了旧的 ViewModelStore
        // 此时 ViewModel 依然活着,它手里还抓着 "旧 Activity" 的引用 -> 泄漏
        mViewModelStore = nc.viewModelStore;
    }
}

结论:

  • 流程:Activity A (销毁) -> ViewModelStore (存活) -> Activity A (重建)
  • 泄漏点:如果 ViewModel 内部持有了 Activity A 的引用(this),因为 ViewModelViewModelStore 里没死,所以 Activity A 也死不了 -> 内存泄漏。
相关推荐
m0_71676523几秒前
C++提高编程--STL初识、string容器详解
java·开发语言·c++·经验分享·学习·青少年编程·visual studio
Memory_荒年1 分钟前
马年驯服不稳定服务:Resilience4j 容错救星驾到!
java·后端
卓怡学长2 分钟前
m278基于web的在线教学质量评价系统
java·数据库·spring·tomcat·maven·intellij-idea
程序员buddha2 分钟前
Java面试八股文Redis篇
java·redis·面试
楼田莉子3 分钟前
高并发内存池项目:内存池性能分析及其优化
开发语言·c++·后端·学习
rannn_1115 分钟前
【Redis|实战篇4】黑马点评|分布式锁
java·数据库·redis·分布式·后端
wapicn997 分钟前
智能识别技术在生活服务领域的落地应用与前景展望
java·c++·人工智能·python·php
是翔仔呐10 分钟前
第6章 UART串口通信!掌握单片机与外界的双向数据通道,实现跨设备交互
c语言·开发语言·单片机·嵌入式硬件·gitee
带娃的IT创业者11 分钟前
从本地开发到 PyPI发布:WeClaw 的 Python 包标准化之旅
开发语言·python
2201_7586426412 分钟前
自定义内存检测工具
开发语言·c++·算法