一次Android Fragment内存泄露的bug解决记录|Fragment not attach to an Activity
Bug描述
前些天出现了一个 bug。Activity 页面里放了一个 ViewPager2,其中的每一页是一个 Fragment。其中第一页的 Fragment 实现了一个监听器,当事件发生和首次添加到监听器管理者 listener manager 时,manager 会通知所有监听者,监听器的回调需要用到当前 Activity 实现一些逻辑。但是在调用requireActivity()
获取 activity 时,页面偶尔会发生 crash,报错提示Exeception: Fragment not attach to an Activity
。
因为是偶现的,于是排先了两个小时的原因😭,在这里记录下来,或许能给大家提供些经验。
场景复现
页面如图一所示
图二 页面布局
简化代码如下
class MyActivity:AppCompatActivity(){...overide fun onCreate(savedInstanceState: Bundle?){super.onCreate(savedInstanceState)// 添加Fragmentval myList = mutableListOf<MyFragment>()myList.add(MyFragment())myList.add(MyFragment())myList.add(MyFragment())val myAdapter = ViewPagerTwoAdapter() // 一个继承了FragmentStateAdapterval viewPager = findViewById(R.id.vp)viewPager.adapter = myAdapter...}...}class MyFragment:Fragment(), MyChangeListenr{...override fun onStart(){super.onStart()myChangeListenerManager.add(this) // 当事件发生和首次添加时,manager会通知所有监听者}override fun doOnChange(){val act = requireActivity() // 这行代码报错not attach to an activity,偶现...}...}interface MyChangeListenr{fun doOnChange()
}
其实看到这里,相信大家已经觉得有些不对劲了,在 Fragment 的 onStart() 函数中,把此 Fragment 作为监听器添加到了 manager 中,但是没有发现相关的 remove,也许就是因为这个,manager 始终持有此 Fragment的 引用,导致发生了内存泄露。但我们还需要一些证据。
分析调试
刚开始遇到这个bug时,我首先想到的:会不会是事件发生的时机正好处在Fragment拿不到Activity的生命周期?于是我再去复习 Fragment 的生命周期。
Fragment的主要生命周期方法依次是onAttach
、onCreate
、onCreateView
、onViewCreated
、onStart
、onResume
、onPause
、onStop
、onDestroyView
、onDestory
、onDetach
。
在 onAttach 执行之后就可以通过 requireActivity 获取 Fragment 所在的 Activity 了,上面我们的时间监听是在 onStart 里才添加的,onStart之后早已可以获取 Activity。接下来通过添加日志,再次检查事件的触发时机。
class MyFragment:Fragment(), MyChangeListenr{...overide fun onAttach(context: Context){super.onAttach(context)Log.d("lxl","onAttach")}override fun onCreate(savedInstanceState: Bundle?){super.onCreate(savedInstanceState)Log.d("lxl","onCreate")}override fun onCreateView(inflater: LayoutInflater, container: ViewGroup?, savedInstanceState: Bundle?){super.onCreateView(inflater, container, savedInstanceState)Log.d("lxl","onCreateView")}override fun onViewCreated(view: View, savedInstanceState: Bundle?){super.onViewCreated(view, savedInstanceState)Log.d("lxl","onViewCreated")}override fun onStart(){super.onStart()Log.d("lxl","onViewCreated")myChangeListenerManager.add(this) // 当事件发生和首次添加时,manager会通知所有监听者}override fun onResume(){super.onResume()Log.d("lxl","onResume")}override fun onPause(){super.onPause()Log.d("lxl","onPause")}override fun onDestoryView(){super.onDestoryView()Log.d("lxl","onDestoryView")}override fun onDestry(){super.onDestry()Log.d("lxl","onDestry")}override fun onDetach(){super.onDetach()Log.d("lxl","onDetach")}override fun doOnChange(){if(isAdd){ // Fragment里的判断是否attach到Activity的方法Log.d("lxl","已绑定")}else{Log.d("lxl","未绑定")}val act = requireActivity() // 这行代码报错not attach to an activity,偶现...}...}
我们尝试运行,第一次进入页面,显示如下
onAttach
onCreate
onCreateView
onViewCreated
onStart
onResume
已绑定
关闭再打开Activity,第二次进入页面crash,显示如下
onAttach
onCreate
onCreateView
onViewCreated
onStart
onResume
未绑定
已绑定
关闭再打开Activity,第三次进入页面crash,显示如下
onAttach
onCreate
onCreateView
onViewCreated
onStart
onResume
未绑定
未绑定
已绑定
可以看出,之后的每一次进入都会打印出来一个未绑定,这说明有多个 Fragment 都收到了监听,那么这些 Fragment 是从哪来的呢?原来是前面退出的 Activity 的 Fragment 没有被释放掉,仍然处在manager 的监听器列表里,这是一种内存泄露。所以我们在不使用 Fragment 时需要移除监听,代码如下
class MyFragment:Fragment(), MyChangeListenr{...override fun onStart(){super.onStart()myChangeListenerManager.add(this) // 当事件发生和首次添加时,manager会通知所有监听者}override fun onStop(){super.onStop()myChangeListenerManager.remove(this) // 当Fragment消失,需要移除监听。防止持有引用,仍然被通知,这是一种内存泄露。}override fun doOnChange(){val act = requireActivity() // 这行代码报错not attach to an activity,偶现...}...}
留下几个疑惑有待分析
- Fragment 被 manager 持有引用无法释放,那么 Fragment 会不会持有 Activity 的引用,导致Activity 无法释放?在这里是 ViewPager2 把 Fragment 和 Activity 绑定,理论上 ViewPager2 会把两者的绑定去掉,可以去看 ViewPager2 源码和发生 bug 时任务栈的情况
- Fragment 的 Stop 函数的执行时机与可见性的关系
想说的话
- 内存泄露是一个新手听了感到棘手的问题,但实际上产生的原因可能很简单,不用生畏。
- 解决 bug 可以通过形式上的合规,修改不规范的地方,这样工作效率更快。如果有余力的情况下可以分析一下 bug 产生的原因,从根本上避免能积累更多经验,避免头痛医头脚痛医脚。
- 学习 Android / 移动端 很开心,请各位大佬批评指正,也可以交个朋友互相交流!
相关文章:

一次Android Fragment内存泄露的bug解决记录|Fragment not attach to an Activity
Bug描述 前些天出现了一个 bug。Activity 页面里放了一个 ViewPager2,其中的每一页是一个 Fragment。其中第一页的 Fragment 实现了一个监听器,当事件发生和首次添加到监听器管理者 listener manager 时,manager 会通知所有监听者࿰…...
基于深度学习的交通标志识别系统
基于深度学习的交通标志识别系统 项目简介 本项目实现了一个基于深度学习的交通标志识别系统,使用卷积神经网络(CNN)对交通标志图像进行分类识别。系统包含数据预处理、模型训练与评估、结果可视化和用户交互界面等模块。 数据集 项目使用德国交通标志识别基准数…...

LVGL图像导入和解码
LVGL版本:8.1 概述 在LVGL中,可以导入多种不同类型的图像: 经转换器生成的C语言数组,适用于页面中不常改变的固定图像。存储系统中的外部图像,比较灵活,可以通过插卡或从网络中获取,但需要配置…...
Vite Proxy配置详解:从入门到实战应用
Vite Proxy配置详解:从入门到实战应用 一、什么是Proxy代理? Proxy(代理)是开发中常用的解决跨域问题的方案。Vite内置了基于http-proxy的代理功能,可以轻松配置API请求转发。 二、基础配置 在vite.config.js中配置…...
oracle goldengate非并行进程转换为并行进程
oracle goldengate非并行进程转换为并行进程 在上一期的文章中写道了直接创建并行进程的方式对大事务进行分解,这对于新建立同步进程的时候提前规划是很有帮助的,但是如果对已经进行了同步的进程重新建立需要耗时比较长,Oracle提供了非并行进…...
VBA将PDF文档内容逐行写入Excel
VBA是无法直接读取PDF文档的,但结合上期我给大家介绍了PDF转换工具xpdf-tools-4.05,先利用它将PDF文档转换为TXT文档,然后再将TXT的内容写入Excel,这样就间接实现了将PDF文档的内容导入Excel的操作。下面的代码将向大家演示如何实…...

project从入门到精通(五)
目录 创建资源的基本信息 在project中创建资源工作表 编辑信息详解 最大单位 标准费率与加班费率 每次使用成本 成本累算 基准日历 三类资源工作表的总结——不同的资源必须要设置的属性 除了资源名称是必须设置的之外,剩余的资源的可设置选项如下图所…...

第3.2.3节 Android动态调用链路的获取
3.2.3 Android App动态调用链路 在Android应用中,动态调用链路指的是应用在运行时的调用路径。这通常涉及到方法调用的顺序和调用关系,特别是在应用的复杂逻辑中,理解这些调用链路对于调试和性能优化非常重要。 1,动态调用链路获…...

亿级流量系统架构设计与实战(六)
微服务架构与网络调用 当某个业务从单体服务架构转变为微服务架构后,多个服务之间会通过网络调用形式形成错综复杂的依赖关系。 在微服务架构中 , 一个微服务正常工作依赖它与其他微服务之间的多级网络调用。 网络是脆弱的 , RPC 请求有较大的概率会遇到超时 、 抖动 、 断…...

浅聊find_package命令的搜索模式(Search Modes)
背景 find_package应该算是我们使用最多的cmake命令了。但是它是如何找到上游库的.cmake文件的? 根据官方文档,整理下find_package涉及到的搜索模式。 搜索模式 find_package涉及到的搜索模式有两种:模块模式(Module mode)和配置模式(Conf…...
开发搭载OneNet平台的物联网数据收发APP的设计与实现
一、开发环境与工具准备 工具安装 下载HBuilderX开发版(推荐使用开发版以避免插件兼容性问题)安装Node.js和npm(用于依赖管理及打包)配置Android Studio(本地打包需集成离线SDK)项目初始化 创建uni-app项目,选择“默认模板”或“空白模板”安装必要的UI库(如uView或Van…...

【LLaMA-Factory】使用LoRa微调训练DeepSeek-R1-Distill-Qwen-7B
【LLaMA-Factory】使用LoRa微调训练DeepSeek-R1-Distill-Qwen-7B 本地环境说明禁用开源驱动nouveau安装nvidia-smi安装Git环境安装Anaconda(conda)环境下载DeepSeek-R1-Distill-Qwen-7B模型安装LLaMA-Factory下载LLaMA-Factory安装LLaMA-Factory依赖修改环境变量安装deepspeedA…...
sh脚本多卡顺序执行训练文件
常规的单机多卡训练脚本一般为 python -m torch.distributed.run --nproc_per_node 2 train.py 上述脚本采用 2 张显卡训练 采用sh脚本,单次顺序执行多个多卡训练文件 例如 train1.py train2.py 特点:在执行完 train1.py之后再执行train2.py文件 …...

使用lldb查看Rust不同类型的结构
目录 前言 正文 标量类型 复合类型——元组 复合类型——数组 函数 &str struct 可变数组vec Iter String Box Rc Arc RefCell Mutex RwLock Channel 总结 前言 笔者发现这个lldb挺好玩的,可以查看不同类型的结构,虽然这好像是C的东…...

【Linux】线程POSIX信号量
目录 1. 整体学习思维导图 2. 信号量的概念 3. 基本接口 4. 基于环形队列的生产者消费者模型(信号量) 1. 整体学习思维导图 2. 信号量的概念 POSIX信号量和SystemV信号量作用相同,都是用于同步操作,达到无冲突的访问共享资源目的。但 POSIX可以用于线…...
WPF中如何自定义控件
WPF自定义控件简化版:账户菜单按钮(AccountButton) 我们以**“账户菜单按钮”为例,用更清晰的架构实现一个支持标题显示、渐变背景、选中状态高亮**的自定义控件。以下是分步拆解: 一、控件核心功能 我们要做一个类似…...
大模型MCP更高效的通信:StreamableHTTP协议
随着大语言模型(LLMs)的飞速发展,模型与应用之间的通信效率和灵活性变得至关重要。Model Context Protocol (MCP) 作为专为模型交互设计的协议,一直在不断进化以满足日益增长的需求。近期,MCP引入了一个令人振奋的新特…...
防火墙在网络安全体系中的核心作用与原理
防火墙在网络安全体系中的核心作用与原理 一、核心作用解析 1. 访问控制中枢 功能维度实现方式典型场景黑白名单控制基于IP/端口/协议的规则过滤限制外部IP访问财务系统,仅开放VPN端口权限分级用户组策略映射(如AD集成)禁止普通员工访问核心…...

MySQL事务和JDBC中的事务操作
一、什么是事务 事务是数据库操作的最小逻辑单元,具有"全有或全无"的特性。以银行转账为例: 典型场景: 从A账户扣除1000元 向B账户增加1000元 这两个操作必须作为一个整体执行,要么全部成功,要么全部失败…...

每日脚本学习5.10 - XOR脚本
xor运算的简介 异或就是对于二进制的数据可以 进行同0异1 简单的演示 : 结果是 这个就是异或 异或的作用 1、比较两数是否相等 2、可以进行加密 加密就是需要key 明文 :0b010110 key : 0b1010001 这个时候就能进行加密 明文 ^ key密文 还有这个加密比…...

【编译原理】总结
核心 闭包,正则闭包 产生式(规则) 文法 G[S](,,P,S) 一组规则的集合 :非终结符 :终结符 P:产生式 S:开始符号 推导 归约 规范(最右ÿ…...

docker创建一个centOS容器安装软件(以宝塔为例)的详细步骤
备忘:后续偶尔忘记了docker虚拟机与宿主机的端口映射关系,来这里查看即可: docker run -d \ --name baota \ --privilegedtrue \ -p 8888:8888 \ -p 8880:80 \ -p 8443:443 \ -p 8820:20 \ -p 8821:21 \ -v /home/www:/www/wwwroot \ centos…...

OpenVLA:开源的视觉-语言-动作模型
1. 简介 让我们先来介绍一下什么是OpenVLA,在这里: https://openvla.github.io/ 可以看到他们的论文、数据、模型。 OpenVLA 是一个拥有 70亿参数的开源 **视觉-语言-动作(VLA)**模型。它是在 Open X-Embodiment 数据集 中的 97万…...

Matlab/Simulink的一些功能用法笔记(4)
水一篇帖子 01--MATLAB工作区的保护眼睛颜色设置 默认的工作区颜色为白色 在网上可以搜索一些保护眼睛的RGB颜色参数设置 在MATLAB中按如下设置: ①点击预设 ②点击颜色,点击背景色的三角标符号 ③点击更多颜色,找到RGB选项 ④填写颜色参数…...
【比赛真题解析】混合可乐
这次给大家分享一道比赛题:混合可乐。 洛谷链接:U561549 混合可乐 【题目描述】 Jimmy 最近沉迷于可乐中无法自拔。 为了调配出他心目中最完美的可乐,Jimmy买来了三瓶不同品牌的可乐,然后立马喝掉了一些(他实在是忍不住了),所以 第一瓶可口可乐最大容量为 a 升,剩余 …...

Elasticsearch:我们如何在全球范围内实现支付基础设施的现代化?
作者:来自 Elastic Kelly Manrique SWIFT 和 Elastic 如何应对基础设施复杂性、误报问题以及日益增长的合规要求。 金融服务公司在全球范围内管理实时支付方面面临前所未有的挑战。SWIFT(Society for Worldwide Interbank Financial Telecommunication -…...

matlab介绍while函数
MATLAB 中的 while 语句介绍 在 MATLAB 中,while 语句是一种循环结构,用于在满足特定条件时反复执行一段代码块。与 for 循环不同,while 循环的执行次数是动态的,取决于循环条件是否为真。 语法 while condition% 循环体代码 e…...

如何解决 PowerShell 显示 “此系统上禁用了脚本运行” 的问题
在 Windows 11 或 10 的 PowerShell 中运行脚本时,你可能会遇到一个错误,提示系统上禁用了脚本运行。这是一种安全功能,而不是系统问题,旨在防止可能有害的脚本自动运行。然而,如果你需要运行脚本来完成某些任务,或者你在系统上做了软件开发或测试的环境,那么你需要在 P…...

深入浅出之STL源码分析4_类模版
1.引言 我在上面的文章中讲解了vector的基本操作,然后提出了几个问题。 STL之vector基本操作-CSDN博客 1.刚才我提到了我的编译器版本是g 11.4.0,而我们要讲解的是STL(标准模板库),那么二者之间的关系是什么&#x…...
探索科技的前沿动态:科技爱好者周刊
探索科技的前沿动态:科技爱好者周刊 在信息爆炸的时代,我们每时每刻都被新技术、新理念包围。而如何在这纷繁复杂的信息中找到对自己有价值的内容,成了一大挑战。今天,我们要介绍的是一个宝贵的资源——科技爱好者周刊,它致力于为科技爱好者提供优质的科技资讯,每周五发…...