Android中常见的那些内存泄漏——【问题分析+方案】

news2024/9/22 5:33:55

1.静态Activity(Activity上下文Context)和View

静态变量Activity和View会导致内存泄漏,在下面代码中对Activity的Context和TextView设置为静态对象,从而产生内存泄漏;

public class MemoryTestActivity extends AppCompatActivity {
 
    private static Context context;
    private static TextView textView;
 
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_memory_test);
        context = this;
        textView = new TextView(this);
    }
}
12345678910111213

因为context和textView的实例的生命周和应用的生命一样,而他们持有当前Activity(MemoryTestActivity)的引用,一旦MemoryTestActivity销毁,而他的引用一直持有,就不会被回收,所以产生内存泄漏了;

1.1借助Android Profiler分析内存泄漏

测试页面关闭以后MemoryTestActivity实例是否全部销毁:

借助Android Profiler可以查看MemoryTestActivity实例有三个(Alloc Count实例数量为3)-不断打开关闭MemoryTestActivity页面,由于关闭MemoryTestActivity不会垃圾回收不会立即执行,为了测试点击强制垃圾回收,我们发现会有一个MemoryTestActivity(@318435616))没有被回收,因为context和textview一直持有MemoryTestActivity(@318435616))实例的引用;

点击Dump Java heap导出堆分配

点击Force Garbage Collection 强制垃圾回收

点击Dump Java heap 导出堆分配

2.单例造成的内存泄漏

Android的单例模式是开发中经常使用的模式,使用不恰当可能导致内存泄漏;单例的生命周期和应用的生命周期一样,也就是单例持有必须是和应用生命周期一样的对象,不能持有和应用生命周期不一致的对象例如:Activity(Context)上下文:

public class TestManager {
 
    private static TestManager manager;
    private Context context;
 
    private TestManager(Context context) {
        this.context = context;
    }
 
    /**
     * 如果传入的context是activity,service的上下文,会导致内存泄漏
     * 原因是我们的manger是一个static的静态对象,这个对象的生命周期和整个app的生命周期一样长
     * 当activity销毁的时候,我们的这个manger仍然持有者这个activity的context,就会导致activity对象无法被释放回收,就导致了内存泄漏
     */
    public static TestManager getInstance(Context context) {
        if (manager == null) {
            manager = new TestManager(context);
        }
        return manager;
    }
}
123456789101112131415161718192021

2.1借助Android Profiler分析内存泄漏

测试反复打开关闭MemoryTestActivity页面,最终停留在MemoryTestActivity页面是否只保留一个实例:

测试步骤:

a.打开MemoryTestActivity,然后关闭;

b.打开MemoryTestActivity,然后停留在MemoryTestActivity页面;

点击Force Garbage Collection 强制垃圾回收

点击Dump Java heap 导出堆分配

会发现第一次打开MemoryTestActivity时创建的实例没有销毁,由于TestManager单例持有MemoryTestActivity引用,TestManager单例生命周期和应用的生命周期一样,所以直到应用的生命周期结束时,TestManager单例持有MemoryTestActivity引用才会被销毁,下图是TestManager单例持有MemoryTestActivity引用;

2.2解决方法

修改TestManager单例模式使用的上下文Context,TestManager单例模式引用ApplicationContext,TestManager单例模式和应用生命周期一样,ApplicationContext和应用的生命周期是一样,这样不会出现内存泄漏;

public class TestManager {
 
    private static TestManager manager;
    private Context context;
 
    private TestManager(Context context) {
        this.context = context;
    }
    //正确写法
    public static TestManager getInstance(Context context) {
        if (manager == null) {
            manager = new TestManager(context.getApplicationContext());
        }
        return manager;
    }
 
}
1234567891011121314151617

3.线程造成的内存泄漏

匿名线程内部类会隐式引用Activity,当执行耗时任务时,一直隐式引用Activity,当Activity关闭时,由于匿名线程内部类会隐式引用Activity无法及时回收;

public class MemoryTestActivity extends AppCompatActivity {
 
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_memory_test);
 
        anonymousInnerClass();
    }
    
    //匿名内部类持有MemoryTestActivity实例引用
    public void anonymousInnerClass(){
        new Thread(){
            @Override
            public void run() {
                //执行异步处理
                SystemClock.sleep(120000);
            }
        }.start();
 
//        new AsyncTask<Void, Void, Void>(){
//            @Override
//            protected Void doInBackground(Void... voids) {
//                //执行异步处理
//                SystemClock.sleep(120000);
//                return null;
//            }
//        }.execute();
    }
}
123456789101112131415161718192021222324252627282930

3.1借助Android Profiler分析内存泄漏

测试MemoryTestActivity实例由于耗时匿名线程内部类引用导致Activity一直无法回收,看如下图Activity打开以后一直无法回收;

当耗时匿名线程内部类执行完成以后MemoryTestActivity实例才会回收;

3.2解决方法

修改Thread和AsyncTask匿名内部类为静态类,解除Activity隐式引用,MemoryTestActivity销毁时要及时取消异步任务staticAsyncTask.cancel(true),防止异步任务执行完成更新销毁MemoryTestActivity实例的UI;

public class MemoryTestActivity extends AppCompatActivity {
 
    private StaticThread staticThread;
    private StaticAsyncTask staticAsyncTask;
    
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_memory_test);
 
//       staticThread = new StaticThread();
//       staticThread.start();
       
       staticAsyncTask = new StaticAsyncTask(this);
       staticAsyncTask.execute();
    }
 
    private static class StaticThread  extends Thread{
        @Override
        public void run() {
            //执行异步处理
            SystemClock.sleep(120000);
        }
    }
 
    private static class StaticAsyncTask extends AsyncTask<Void, Void, Void>{
        private WeakReference<Context> weakReference;
 
        public StaticAsyncTask(Context context){
            weakReference = new WeakReference<Context>(context);
        }
 
        @Override
        protected Void doInBackground(Void... voids) {
            //执行异步处理
            SystemClock.sleep(120000);
            return null;
        }
 
        @Override
        protected void onPostExecute(Void aVoid) {
            super.onPostExecute(aVoid);
 
            MemoryTestActivity activity = (MemoryTestActivity)weakReference.get();
            if(activity != null){
                //异步任务执行完成,执行UI处理
            }
        }
    }
 
    @Override
    protected void onDestroy() {
        super.onDestroy();
        staticAsyncTask.cancel(true);
    }
}
1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556

借助Android Profiler可以查看返回打开关闭MemoryTestActivity页面,看MemoryTestActivity实例是否销毁:

可以参考匿名线程引用类的测试图对比一下,不存在MemoryTestActivity实例,MemoryTestActivity实例正常被销毁了,没有因为静态异步线程类执行耗时操作而导致MemoryTestActivity不销毁出现内存泄漏的问题;

4.非静态内部类创建静态实例造成的内存泄漏

有时候我们可能频繁启动Activity,为了避免重复创建 相同的资源呢,会出现如下写法:

public class MemoryTestActivity extends AppCompatActivity {
 
 
    private static TestResource testResource;
 
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_memory_test);
 
        testResource = new TestResource();
    }
 
    class TestResource{
       //资源类
    }
 
}
123456789101112131415161718

这样就在Activity内部创建了一个非静态内部类的单例,每次启动Activity时都会使用该单例的数据,这样虽然避免重复创建,不过这种写法会造成内存泄漏,因为非静态内部类默认会持有外部类的引用,而又使用了该非静态内部类创建了一个静态实例,该实例的生命周期和应用一样长,这就导致了该静态实例一直持有该Activity的引用,导致Activity的内存资源不能正常回收;

4.1借助Android Profiler分析内存泄漏

测试打开关闭MemoryTestActivity页面,MemoryTestActivity实例是否正常销毁;

4.2解决方法

将该内部类设为静态内部类或将内部类抽象出来封装一个单例,如果需要使用Context,请使用ApplicationContext;

5.Handler造成的内存泄漏

Handler的使用造成的内存泄漏问题是比较常见的,平时处理网络任务或者封装一些请求回调等api都应该会借助Handler处理,对于Handler的使用代码不规范可能会造成内存泄漏,如下示例:

public class MemoryTestActivity extends AppCompatActivity {
 
 
    private Handler mHandler = new Handler(){
        @Override
        public void handleMessage(Message msg) {
            //处理UI显示
        }
    };
 
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_memory_test);
 
        loadData();
    }
 
    //loadData()方法在子线程中执行
    private void loadData() {
        //子线程执行网络请求request
        Message message = Message.obtain();
        //模拟线程延迟120秒发送Message
        mHandler.sendMessageDelayed(message, 120000);
//        mHandler.sendMessage(message);
    }
 
 
}
1234567891011121314151617181920212223242526272829

这种创建Handler的方式可能造成内存泄漏,由于mHandler是Handler的非静态匿名内部类的实例,所以它持有外部类Activity引用,我们知道消息队列是在一个Looper线程中不断轮询处理消息,那么当这个Activity退出时,消息队列还有未处理的消息或者正在处理的消息(例如上面的例子,子线程中处理耗时任务,还没有执行完毕,activity就退出销毁),而消息队列中Message持有mHandler实例引用,mHander又持有Activity的引用,所以导致Activity的内存无法及时回收,引发内存泄漏;

5.1借助Android Profiler分析内存泄漏

借助Android Profiler可以查看返回打开关闭MemoryTestActivity页面,看MemoryTestActivity实例是否销毁:

通过Android Profiler可以发现打开关闭MemoryTestActivity页面,发现MemoryTestActivity实例由于mHandler一直持有外部类Activity引用没有销毁;

5.2解决方法

public class MemoryTestActivity extends AppCompatActivity {
 
    private Handler handler = new StaticHandler(this);
 
    private static class StaticHandler extends Handler{
 
        WeakReference<Context> weakReference =  null;
 
        public StaticHandler(Context context){
            weakReference = new WeakReference<Context>(context);
        }
 
        @Override
        public void handleMessage(Message msg) {
            //处理UI显示
            MemoryTestActivity  activity =(MemoryTestActivity)weakReference.get();
            if(activity !=  null){
 
            }
        }
    }
 
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_memory_test);
 
        loadData();
    }
 
    //loadData()方法在子线程中执行
    private void loadData() {
        //子线程执行网络请求request
        Message message = Message.obtain();
        //模拟线程延迟120秒发送Message
        handler.sendMessageDelayed(message, 120000);
//        mHandler.sendMessage(message);
    }
 
    @Override
    protected void onDestroy() {
        super.onDestroy();
        handler.removeCallbacksAndMessages(null);
//        handler.removeCallbacks();
//        handler.removeMessages();
    }
}
1234567891011121314151617181920212223242526272829303132333435363738394041424344454647

创建一个静态Handler内部类,然后对Handler持有的对象使用弱应用,这样在回收时也可以回收Handler持有的对象,这样避免了Activity泄漏,如果Handler被delay(延迟执行),在Activity的Destroy或者Stop时应该移除消息队列中的消息;

handler.removeCallbacksAndMessages(null);移除消息队列中所有的消息和线程; handler.removeCallbacks();移除消息队列中指定的线程; handler.removeMessages();移除消息队列中指定的消息;

解决方案总结:

1.通过程序逻辑来进行维护

a.在关闭Activity的时候停掉后台线程;线程停掉相当于切断了Handler和外部连接线,Activity自然会被在合适的时候回收;

b.如果Handler被delay延迟的Message持有了引用,那么使用相应的Handler的removeCallbacks()方法,把消息对象从消息队列移除就行;

2.将Handler声明为静态类

a.在Java中,非静态的内部类和匿名内部类都会隐式持有其外部类的引用,静态内部类不会持有外部类的引用。静态类不持有外部类的对象,所以你的Activity可以随意被回收;由于Handler不在持有外部类的对象的引用,导致程序不允许你在Handler中操作Activity中的对象了,所以你需要在Handler中增加一个对Activity的弱引用(WeakReference);

6.动画

在属性动画中有一类无限循环动画,如果在Activity中播放这类动画并且在onDestroy()中没有去停止动画,那么动画会一直播放下去,这时候Activity会被View所持有,从而导致Activity无法被释放。解决此类问题要在onDestroy()方法中去调用objectAnimator.cancel()来停止动画;

public class MemoryTestActivity extends AppCompatActivity {
 
    private TextView textView;
    private ObjectAnimator objectAnimator;
 
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_memory_test);
 
        textView = (TextView)this.findViewById(R.id.textView2);
        objectAnimator = ObjectAnimator.ofFloat(textView, "rotation", 0, 360);
        objectAnimator.setRepeatCount(ValueAnimator.INFINITE);
        objectAnimator.start();
    }
 
    @Override
    protected void onDestroy() {
        super.onDestroy();
 
    }
}
12345678910111213141516171819202122

6.1借助Android Profiler分析内存泄漏

借助Android Profiler可以查看返回打开关闭MemoryTestActivity页面,看MemoryTestActivity实例是否销毁:

由于未在onDestroy()方法中去调用objectAnimator.cancel()来停止动画,执行动画的View一直引用Activity,导致Activity无法销毁;

6.2解决办法

在onDestroy()方法中去调用objectAnimator.cancel()来停止动画;

7.第三方库使用不当

对于EventBus,RxJava等一些第三方开源框架的使用,若是Activity销毁之前没有进行解除订阅会导致内存泄漏;

8资源未关闭造成的内存泄漏

对于使用了BraodcastReceiver,ContentObserver,File,Cursor,Stream,Bitmap等资源的使用,应该在Activity销毁时及时关闭或者注销,否则这些资源将不会被回收,造成内存泄漏。

9.Java内存监测工具

9.1Android Profiler

在Android Studio中View-Tool Windows-Android Profiler可以查看工具自带的内存分析工具;

9.2leakcanary

在app的build.gradle中添加如下代码,不需要其他处理;

dependencies {
    // debugImplementation because LeakCanary should only run in debug builds.
    debugImplementation 'com.squareup.leakcanary:leakcanary-android:2.4'
}
1234

查看LogCat控制台,输出如下日志则说明leakcanary已经启动:

D LeakCanary: LeakCanary is running and ready to detect leaks
​
 11

添加好了依赖后,当有内存泄漏出现时,在通知栏会有提示,点击进入后,就会看到具体的报错点。leakcanary 的基本情况是,他在debug下,是会去检测你的应用内存泄漏情况,而在release版下,不会去检测的。

以上就是Android开发中比较常见的内存泄漏问题以及解决方法;如果想要更加深入学习Android的性能优化,或者Android技术进阶大家可以参考《Android核心技术手册》这个资料文件。里面记录了大部分的Android开发技术点,供你学习参考!

文末

1、Handler持有的引用最好使用弱引用,在Activity被释放的时候要记得清空Message,取消Handler对象的Runnable;

2、非静态内部类、非静态匿名内部类会自动持有外部类的引用,为避免内存泄露,可以考虑把内部类声明为静态的;

3、对于生命周期比Activity长的对象,要避免直接引用Activity的context,可以考虑使用ApplicationContext;

4、广播接收器、EventBus等的使用过程中,注册/反注册应该成对使用;

5、不再使用的资源对象Cursor、File、Bitmap等要记住正确关闭;

6、集合里面的东西、有加入就应该对应有相应的删除。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/33074.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

[附源码]SSM计算机毕业设计健身健康规划系统JAVA

项目运行 环境配置&#xff1a; Jdk1.8 Tomcat7.0 Mysql HBuilderX&#xff08;Webstorm也行&#xff09; Eclispe&#xff08;IntelliJ IDEA,Eclispe,MyEclispe,Sts都支持&#xff09;。 项目技术&#xff1a; SSM mybatis Maven Vue 等等组成&#xff0c;B/S模式 M…

noexcept说明符/运算符

一、noexcept说明符 1、语法 &#xff08;1&#xff09;noexcept 与 noexcept(true) 相同 &#xff08;2&#xff09;noexcept&#xff08;表达式&#xff09; 如果 表达式 求值为 true&#xff0c;那么声明函数不会抛出任何异常。 &#xff08;3&#xff09;throw() //c1…

Ubuntu配置FTP服务

参考目录1.安装FTP服务器软件2.配置FTP服务3.Ubuntud登录ftp服务器4.windows下通过cuteFTPlianjei1.安装FTP服务器软件 (1) FTP文件传送协议(File Transfer Protocol&#xff0c;简称FTP)&#xff0c;是一个用于从一台主机到另一台主机传输文件的协议。 (2&#xff09;Linux下有…

Jetpack 之 LiveData 实现事件总线

事件总线相信大家很多时候都会用到&#xff0c;那大家常用的也就是常青树 EventBus&#xff0c;以及 RxJava 流行起来的后起之秀 RxBus。它们的使用方式都差不多&#xff0c;思想也都是基于观察者模式&#xff0c;正好 LiveData 的核心思想也是观察者模式&#xff0c;因此我们完…

做Android 开发这么久,还不明白 Android Framework 知识重要性?

Framework作为Android的框架层&#xff0c;为App提供了很多API调用&#xff0c;但很多机制都是Framework包装好后直接给App用的&#xff0c;如果不懂这些机制的原理&#xff0c;就很难在这基础上进行优化。 从做Android的第一天起&#xff0c;你一定听过无数次关于Framework的…

计算机音乐-乐理知识(1)

一、节拍 节拍&#xff08;Beat/Meter&#xff09;&#xff0c;是一个衡量节奏的单位&#xff0c;在音乐中&#xff0c;有一定强弱分别的一系列拍子在每隔一定时间重复出现。如 2 / 4 、 4 / 4 、 3 / 4 拍等。节拍&#xff0c;乐曲中表示固定单位时值和强弱规律的组织形式。 …

测试员工作三年后的工资对比,没达到这个数的都属于拖后腿了

“毕业三年的薪资是职场阶段的一个分水岭。” 不知什么时候开始&#xff0c;这句话深刻的引入了所有打工人的心中&#xff0c;程序员们自然也不例外。 事实上&#xff0c;这句话说的并不无道理&#xff0c;毕业的三年&#xff0c;不仅是学生到职场人身份上的一个转变&#xf…

初阶数据结构学习记录——아홉 二叉树和堆(2)

接着上一篇 之前写过一些关于堆的代码&#xff0c;向下调整&#xff0c;向上调整算法&#xff0c;以及常用的几个函数。这一篇继续完善堆&#xff0c;难度也会有所上升。先来看上一篇文末提到的创建堆算法。 首先要有空间&#xff0c;要有数据&#xff0c;之后再形成堆。我们…

9.5 利用可执行内存挑战DEP

目录 一、实验环境 二、实验思路 三、实验代码 四、实验步骤 1、寻找memcpy函数的地址 2、查看内存中可读可写可执行的内存 3、修复EBP 4、保证memcpy的源地址位于shellcode之前 一、实验环境 操作系统&#xff1a;windows 2000 软件&#xff1a;原版OD、VC6.0 二、实…

删除的数据如何恢复?误删了文件怎么恢复

文件的误删除&#xff0c;相信大部分人都经历过。不过因为很多人删除的文件都不算是很重要&#xff0c;所以有与没有并没有太大的区别。但是一旦你删除的文件正是你最近急需的&#xff0c;删除的数据如何恢复&#xff1f;别着急&#xff0c;可以试试以下的几种方法&#xff1a;…

STM32串口详解

实验一&#xff1a;简单的利用串口接收中断回调函数实现数据的返回 关于串口调试助手&#xff0c;还应知道&#xff1a; 发送英文字符需要用一个字符即8位&#xff0c;发送汉字需要两个字符即16位&#xff0c;如上图&#xff0c;发送汉字“姜”实际是发送“BD AA”而发送英文字…

外卖项目06---套餐管理业务开发(移动端的后台代码编辑开发)

菜品展示、购物车、下单 目录 一、导入用户地址簿相关功能代码 90 1.1需求分析 90 1.2数据模型 90 1.3导入功能代码 90 二、菜品展示 91 2.1需求分析 91 2.2商品展示---代码开发---梳理交互过程 92 2.3菜品展示---代码开发---修改DishController的list方法并测试 93 2…

OpenGL原理与实践——核心模式(二):Shader变量、Shader类的封装以及EBO

目录 Shader内的一些关键字 向量 举例&#xff1a;shader之间的数据传输&#xff0c;并实现渐变颜色 举例&#xff1a;C向shader传输数据的过程 代码整理——shader类的封装 加入颜色信息 索引绘制——EBO 整体代码以及渲染结果 Shader内的一些关键字 in&#xff1a;上…

网站被劫持勒索怎么办

互联网出现后的几十年时间里&#xff0c;世界便由一张张网串联了起来&#xff0c;给我们的生活带来了无限的便利。但在互联网飞速发展的同时&#xff0c;恶意网络攻击也随之而来&#xff0c;近年来&#xff0c;互联网攻击事件频发&#xff0c;不法分子利用常见的DDoS攻击、CC攻…

【生成式网络】入门篇(二):GAN的 代码和结果记录

GAN非常经典&#xff0c;我就不介绍具体原理了&#xff0c;直接上代码。 感兴趣的可以阅读&#xff0c;里面有更多变体。 https://github.com/rasbt/deeplearning-models/tree/master/pytorch_ipynb/gan GAN 在 MINIST上的代码和效果 import os # os.chdir(os.path.dirname(_…

springBoot集成websocket实现消息实时推送提醒

在浏览某些网页的时候&#xff0c;例如 WebQQ、京东在线客服服务、CSDN私信消息等类似的情况下&#xff0c;我们可以在网页上进行在线聊天&#xff0c;或者即时消息的收取与回复&#xff0c;可见&#xff0c;这种功能的需求由来已久&#xff0c;并且应用广泛,和pc端web系统待办…

新建anaconda使用jupyter出现的一系列问题

1&#xff0c;运行一段机器学习代码&#xff0c;报缺少h5py的错误. 使用conda install h5py1.8.0 安装无法安装&#xff0c;因为当前环境的python版本是3.9&#xff0c;只能用3.7及以下的版本。无奈只能新建一个conda 环境。 2&#xff0c;新建一个 python3.7的conda 环境。运行…

「风控算法服务平台」高性能在线推理服务设计与实现

本文作者&#xff1a;郁昌存 来自京东科技-风险管理中心 一、背景/目标 1&#xff09; 风控智能化体系建设依赖大量深度学习/机器学习模型进行实时在线的风险识别、智能决策。要求可以将算法模型快速部署为在线服务&#xff0c;供决策引擎调用。 2&#xff09; 风控决策引擎…

文献 | 教师主观幸福感变迁:横断历史研究的视角

Hello&#xff0c;大家好&#xff01; 这里是壹脑云科研圈&#xff0c;我是莹~ 疫情带来的社会经济变化正在改变着我们的求职意向&#xff0c;越来越多的人参与到考公考编的大军中。其中&#xff0c;教师这一职业的稳定性和社会认同度吸引了越来越多的年轻人参加教资考试。 教…

Linux Top 详细介绍,包含task排序

Linux Top 当我们在终端输入 top 命令时&#xff0c;会弹出一个变化的页面&#xff0c;打印出当前系统的大量重要指标&#xff0c;以及很多进程当前的运行情况&#xff1a; 可以看到&#xff0c;top 命令主要是两部分&#xff0c;第一部分为 头部指标&#xff0c;打印的是当…