Android 内存泄漏

news/2025/2/2 9:46:47/

名词解释

内存泄漏:即memory leak。是指内存空间使用完毕后无法被释放的现象,虽然Java有垃圾回收机制(GC),但是对于还保持着引用, 该内存不能再被分配使用,逻辑上却已经不会再用到的对象,垃圾回收器不会回收它们。

内存溢出:即out of memory, 当你要求分配的内存超过了系统给你的内存时, 系统就会抛出out of memory的异常(每个Android能用的内存是有限的) 。比如: 当前应用只剩下4M的空间可用, 但你却加载得到一个需要占用5M空间的图片Bitmap对象, 就会抛出溢出的异常

常见内存泄漏场景&解决方案

1.非静态内部类、匿名类()

非静态内部类会持有外部类的引用,如果非静态内部类的实例是静态的,就会长期的维持着外部类的引用,阻止被系统回收。
解决方案是使用静态内部类

1.1非静态内部类

非静态内部类(non static inner class)和 静态内部类(static inner class)之间的区别。
在这里插入图片描述

如果非静态内部类所创建的实例是静态的其生命周期等于应用的生命周期。非静态内部类默认持有外部类的引用而导致外部类无法释放,最终造成内存泄露。即外部类中持有非静态内部类的静态对象。

public class MainActivity extends Activity {//非静态内部类的静态实例引用public static TestClass testClass = null;@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);//保证非静态内部类的实例只有1个if (testClass == null) {testClass = new TestClass();}}// 非静态内部类private class TestClass {//todo something}
}

当 MainActivity 销毁时,因非静态内部类单例的引用,testClass 的生命周期等于应用的生命周期,持有外部类 MainActivity 的引用,故 MainActivity 无法被 GC 回收,从而导致内存泄漏。

解决方案:
将非静态内部类设置为:静态内部类(静态内部类默认不持有外部类的引用)
该内部类抽取出来封装成一个单例
尽量避免非静态内部类所创建的实例是静态的

1.2 多线程相关的匿名内部类和非静态内部类(继承 Thread 类、实现 Runnable 接口、AsyncTask)

当子线程正在处理任务时,如果外部类销毁, 由于子线程实例持有外部类引用,将使得外部类无法被垃圾回收器(GC)回收,从而造成内存泄露。

public class MainActivity extends Activity {@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);new MyThread().start();}private class MyThread extends Thread {@Overridepublic void run() {//todo someting}}
}

解决方案:

  1. 使用静态内部类的方式,静态内部类不默认持有外部类的引用。
private static class MyThread extends Thread {@Overridepublic void run() {//todo someting}}
  1. 当外部类结束生命周期时(即Activity或Fragment),强制结束线程(onDestroy或onDestroyView)。使得工作线程实例的生命周期与外部类的生命周期同步。
    @Overrideprotected void onDestroy() {super.onDestroy();myThread.interrupt();}

2. Handler内存泄漏(重新理解为什么 Handler 可能导致内存泄露?)

Handler内部message是被存储在MessageQueue中的,MessageQueue中的 Message 持有 Handler 实例的引用,有些message不能马上被处理,存在的时间会很长,导致handler无法被回收,如果handler是非静态的(内部类、匿名内部类),默认持有外部类的引用(如 MainActivity 实例),导致它的外部类无法被回收。
在这里插入图片描述

public class MainActivity extends Activity {private MyHandler myHandler;@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);myHandler = new MyHandler();new Thread() {@Overridepublic void run() {try {//执行耗时操作Thread.sleep(20000);} catch (InterruptedException e) {e.printStackTrace();}//发送消息myHandler.sendEmptyMessage(1);}}.start();}private class MyHandler extends Handler {@Overridepublic void handleMessage(Message msg) {//处理消息事件}}
}

解决方案:

  1. 使用静态内部类+弱引用的方式,保证外部类能被回收。因为弱引用的对象拥有短暂的生命周期,在垃圾回收器线程扫描时,一旦发现了具有弱引用的对象,不管当前内存空间足够与否,都会回收它的内存。
public class MainActivity extends Activity {private MyHandler myHandler;@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);myHandler = new MyHandler(this);new Thread() {@Overridepublic void run() {try {//执行耗时操作Thread.sleep(10000);} catch (InterruptedException e) {e.printStackTrace();}//发送消息myHandler.sendEmptyMessage(1);}}.start();}public void test() {Log.d("TAG", "test");}private static class MyHandler extends Handler {//定义弱引用实例private WeakReference<Activity> reference;//在构造方法中传入需持有的Activity实例public MyHandler(Activity activity) {//使用 WeakReference 弱引用持有 Activity 实例reference = new WeakReference<Activity>(activity);}@Overridepublic void handleMessage(Message msg) {//处理消息事件//调用Activity实例中的方法((MainActivity) reference.get()).test();}}
}
  1. 当外部类结束生命周期时,清空 Handler 内消息队列。
    @Overrideprotected void onDestroy() {if (myHandler!= null) {myHandler.removeCallbacksAndMessages(null);}super.onDestroy();}

3. Context导致内存泄漏

根据场景确定使用Activity的Context还是Application的Context,因为二者生命周期不同,对于不必须使用Activity的Context的场景(Dialog),一律采用Application的Context,单例模式是最常见的发生此泄漏的场景,比如传入一个Activity的Context被静态类引用,导致无法回收

4. 静态View导致泄漏

使用静态View可以避免每次启动Activity都去读取并渲染View,但是静态View会持有Activity的引用,导致无法回收。

解决方案:

  1. 尽量避免 static 成员变量引用资源耗费过多的实例(如 Context),若需引用 Context,则尽量使用Applicaiton的 Context。
  2. 使用弱引用(WeakReference) 代替强引用持有实例。
  3. 在Activity销毁的时候将静态View设置为null

5.资源对象未关闭导致

对于资源若在 Activity 销毁时无及时关闭 / 注销这些资源,则这些资源将不会被回收,从而造成内存泄漏。
如广播、文件、Bitmap、数据库等使用

//对于广播BroadcastReceiver:注销注册
unregisterReceiver(broadcastReceiver);//对于文件流File:关闭流
inputStream / outputStream.close();//对于数据库游标cursor:使用后关闭游标
cursor.close();//对于图片资源Bitmap:Android分配给图片的内存只有8M,若1个Bitmap对象占内存较多,当它不再被使用时,应调用recycle()回收此对象的像素所占用的内存;最后再赋为null 
bitmap.recycle();
bitmap = null;// 对于动画(属性动画),将动画设置成无限循环播放setRepeatCount(ValueAnimator.INFINITE);后
// 在Activity退出时记得停止动画
animator.cancel();

6.监听器未关闭

很多需要register和unregister的系统服务要在合适的时候进行unregister,手动添加的listener也需要及时移除

7.集合中的对象未清理

集合用于保存对象,如果集合越来越大,不进行合理的清理,

8. WebView导致的内存泄漏(目前没有遇到)

WebView只要使用一次,内存就不会被释放,所以WebView都存在内存泄漏的问题。
通常的解决办法是为WebView单开一个进程,使用AIDL进行通信,根据业务需求在合适的时机释放掉

内存泄漏分析工具

lint

lint 是一个静态代码分析工具,同样也可以用来检测部分会出现内存泄露的代码,平时编程注意 lint 提示的各种黄色警告即可。如:
在这里插入图片描述
也可以手动检测,在 Android Studio 中选择 Code->Inspect Code。
在这里插入图片描述
然后会弹出选择检测范围
在这里插入图片描述

点击Ok,等待分析结果
在这里插入图片描述

这个工具除了会检测内存泄漏,还会检测代码是否规范、是否有没用到的导包、可能的bug、安全问题等等。

Memory Profile

Memory Profile 的使用

LeakCanary

LeakCanary

链接

Android内存优化(三)避免可控的内存泄漏
Android应用开发中对Bitmap的内存优化
Android内存优化(五)详解内存分析工具MAT
Android 中内存泄漏的原因分析及解决方案
手把手教你在Android Studio 3.0上分析内存泄漏
RxJava这么好用却容易内存泄漏?解决办法是…
搞定 Android App 的内存泄漏问题
再见,内存泄漏!
关于LiveData可能引发的内存泄漏及优化
LeakCanary纠察内存泄漏后,为什么还是OOM?
Android内存优化(六)LeakCanary使用详解
查内存泄漏神器,LeakCanray原来是这样工作的
LeakCanary源码浅析,内存泄漏检测的好帮手
被问到:如何检测线上内存泄漏,通过 LeakCanary 探究!
Android 内存泄漏检测工具 LeakCanary 的使用
Android 内存泄露分析


http://www.ppmy.cn/news/345944.html

相关文章

码云实战(一)——idea实现将本地的项目推送到码云上

文章目录 前言一、创建本地仓库并关联二、将项目提交本地仓库三、关联远程仓库3.1 创建空白的远程库 四、推送到远程仓库五、验证是否推送成功总结 前言 本系列文章主要记录日常使用中碰到的码云的相关问题 一、创建本地仓库并关联 用IDEA打开项目&#xff0c;在菜单栏点击vc…

HDMI 发送芯片——MS7210

MS7210 是一款 HDMI 发送芯片&#xff0c;兼容 HDMI1.4b 及 HDMI 1.4b 下的视频 3D 传输格式。可以支持的最高分辨率高达 4K30Hz&#xff0c;最高采样率达到 300MHz.MS7210 支持 YUV 和 RGB 之间的色彩空间转换&#xff0c;数字接口支持 YUV 以及 RGB 格式输入。MS7210 的 IIS …

【Centos8】Centos8+使用MegaCLI查看硬件RAID情况

Centos8使用MegaCLI查看LSI厂商硬件RAID情况 前言问题记录报错解决方案 Megacli常用命令【查看raid同步进度】 查看磁盘缓存策略 前言 最近公司服务器做了硬件raid5的服务器出现硬盘故障&#xff0c;需要更换硬盘。与厂商二次确认后买的RAID卡做的硬件Raid5是支持热插拔的。期…

Windows下编译Mitsuba 3

1. 获取源码 打开cmd&#xff0c;打开&#xff08;cd&#xff09;要放置Mitsuba3的文件夹&#xff0c;从Mitsuba 3的github仓库上将其拉到本地&#xff0c;执行&#xff1a; git clone --recursive https://github.com/mitsuba-renderer/mitsuba3 由于Mitsuba3还有一些子模块…

安装ORB-SLAM2纯记录(caffe)

ubuntu20.04 显卡配置3050TI 显卡驱动&#xff1a; NVIDIA-SMI 525.116.04 Driver Version: 525.116.04 cuda:Cuda compilation tools, release 11.0, V11.0.221 eigen:3.4.0 opencv 3.4.9 opencv下载链接 pangolin :高翔SLAM十四讲中的 没有安装最新版本 安装pangolin时候出…

高考落榜,误打误撞学习了软件测试现在月薪30k成为了班上人人羡慕的对象

记得我刚高考结束时&#xff0c;并没有想象中的狂欢&#xff0c;反而是一种处于一种坐立不安的焦虑中&#xff0c;因为那时单纯地认为&#xff1a;这张试卷&#xff0c;将决定我的一生。对于将信仰寄托于高考的学生来说&#xff0c;当网页上高考成绩弹出的一瞬间&#xff0c;世…

【开源项目】昆明~超经典智慧城市智慧昆明CIM/BIM数字孪生可视化项目——开源工程及源码

为了打造更加智能化的城市&#xff0c;智慧昆明借助数字孪生技术&#xff0c;将城市的真实世界实时反映到数字世界中&#xff0c;实现了城市的快速分析和精确预测。更为棒的是&#xff0c;智慧昆明工程及源码均免费赠送&#xff0c;助您的城市更快步入智慧化时代&#xff01; 项…

ArcFaceLoss(人脸检测)

人脸识别常用的损失函数: train_arc.py import torch import torch.nn as nn import torch.utils.data as data import torchvision import matplotlib.pyplot as plt from ArcLoss import Arc from Net import Netdevice torch.device("cuda" if torch.cuda.is_a…