Android常见内存泄漏分析

内存泄漏

内存泄漏就是分配的内存空间没有及时回收导致的。可使用的内存变少,应用变卡,最后内存溢出后应用就会挂掉

内存泄漏的检测

建议阅读Android内存泄漏检测和定位这篇文章,使用里面的检测方法可以轻松的验证本文中的内存泄漏例子

原因

Android内存泄漏大多是因为Activity没有被回收导致的,Activity没有被回收一般分为两种情况

  • 全局的static变量持有Activity的强引用
  • 在Activity生命周期外的线程,持有Activity的强引用

引用类型

推荐阅读Java中四种引用类型,感觉是非常容易理解的

静态变量

类中定义了静态Activity变量,把当前的Activity赋值给静态变量,如果Activity生命周期结束的时候静态变量没有清空,就会导致内存泄漏。static变量是贯穿整个应用的生命周期的,所以被泄漏的Activity就会一直存在于应用的进程中,不会被回收,同样的持有Activity(Context)的静态变量,比如View也是一样的道理

private static Activity sActivity;

@Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main2);
        sActivity = this;
        findViewById(R.id.btn_back).setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View view) {
                finish();
            }
        });

    }

内部类

非静态内部类 和 匿名类 都会潜在的引用它们所属的外部类,但是静态内部类却不会。

private static Test sTest;
    private static Test2 sTest2;

    class Test { }
    static class Test2 { }
    
    private void test() {
        sTest = new Test();
        sTest2=new Test2();
    }
屏幕快照 2019-02-13 下午1.24.04.png-110.3kB

1.如果这个非静态内部类实例内部做了一些耗时的操作,就会导致外围对象不会被回收,从而导致内存泄漏

protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        test();
    }
private void test() {
        new Thread(new Runnable() {
            @Override
            public void run() {
                while (true) {
                    try {
                        Thread.sleep(1000);
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                }
            }
        }).start();
    }

这个匿名内部类会引用这个Activity,内部开了线程做耗时操作,就会导致这个Activity不能被回收

2.结合上面的静态变量,如果静态变量持有非静态内部类的引用,而
非静态内部类引用了该Activity,那就会导致这个Activity不能被回收

private static Test sTest;
    class Test {}
    private void test() {
        sTest = new Test();
    }

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        test();
    }

Test是非静态内部类,sTest是静态变量,注意这个静态变量是在onCreate中初始化的,会持有该Activity的引用,Activity被销毁的时候sTest不置空那么该Activity就无法被回收;

内部类常见情况

上面都说了,非静态内部类会引用所属外部类,这时候如果创建一个内部类,而且持有一个静态变量的引用就容易会引起外部类没法被回收;同样的如果该内部类在子线程做了一些耗时操作,属于在Activity生命周期外的线程,也会导致外部类没法被回收;常见的情况有下面几种

Threads

上面第一个例子已经写过了

TimerTask

匿名内部类嘛,肯定就持有所在Activity的引用,又做耗时操作,肯定内存泄漏

Handler

一个道理,匿名内部类嘛,肯定就持有所在Activity的引用,如果执行postDelayed的时候,Activity被销毁,那么Handler持有的Activity没法被回收,就内存泄漏了,而且里面也有一个匿名内部类Runnable持有Activity

new Handler(){
            @Override
            public void handleMessage(Message msg) {
                super.handleMessage(msg);
            }
        }.postDelayed(new Runnable() {
            @Override
            public void run() {
                
            }
        },10000);

系统服务

这个也很好理解,当你使用系统服务的时候,可以注册监听器,会导致服务持有Context的引用,如果在Activity销毁的时候,没有注销掉监听器,就会导致内存泄漏;

//传感器的监听器注册
SensorManager sensorManager = (SensorManager) getSystemService(SENSOR_SERVICE);
Sensor sensor = sensorManager.getDefaultSensor(Sensor.TYPE_ALL);
sensorManager.registerListener(this, sensor, SensorManager.SENSOR_DELAY_FASTEST);

注册广播接收器也是需要在onDestroy方法里面注销

资源未释放

这个简单,一般就各种流、各种资源没有关闭,集合中对象没清理导致,倒是不容易犯错

静态内部类解决内存泄漏

1.静态内部类不会持有外部类的引用,所以使用静态内部类可以解决以上问题,如果静态内部类里面需要引用外部类,可以通过弱引用的方式来引用;
2.用static的变量引用匿名内部类的实例或将匿名内部类的实例化操作放到外部类的静态方法中

//静态内部类
private static class Myhandler extends Handler {
        private final WeakReference<Activity> mActivity;

        public Myhandler(Activity activity) {
            mActivity = new WeakReference<Activity>(activity);
        }

        @Override
        public void handleMessage(Message msg) {
            super.handleMessage(msg);
            Activity activity = mActivity.get();
            // ...
        }
    }
    
private final Myhandler mMyhandler=new Myhandler(this);

//这样写不会持有外部类的引用
private static final Runnable sRunnable = new Runnable() {
      @Override
      public void run() { /* ... */ }
  };

@Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        mMyhandler.postDelayed(sRunnable, 1000 * 60 * 10);
    }

方法没有问题,但是为什么上面的写法sRunnable没有引用外部类而下面的写法会引用呢,会导致内存泄漏呢;我觉得是因为初始化的位置不同,静态变量和静态类先被初始化了,所以没有外部类的引用;感觉是这样的吧

private static Runnable sRunnable;
 @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        sRunnable = new Runnable() {
            @Override
            public void run() { /* ... */ }
        };
    }

参考文章:

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 157,198评论 4 359
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 66,663评论 1 290
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 106,985评论 0 237
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 43,673评论 0 202
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 51,994评论 3 285
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,399评论 1 211
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,717评论 2 310
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,407评论 0 194
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,112评论 1 239
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,371评论 2 241
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 31,891评论 1 256
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,255评论 2 250
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 32,881评论 3 233
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,010评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,764评论 0 192
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,412评论 2 269
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,299评论 2 260

推荐阅读更多精彩内容

  • 内存管理的目的就是让我们在开发中怎么有效的避免我们的应用出现内存泄漏的问题。内存泄漏大家都不陌生了,简单粗俗的讲,...
    DreamFish阅读 774评论 0 5
  • Android 内存泄漏总结 内存管理的目的就是让我们在开发中怎么有效的避免我们的应用出现内存泄漏的问题。内存泄漏...
    _痞子阅读 1,586评论 0 8
  • 内存管理的目的就是让我们在开发中怎么有效的避免我们的应用出现内存泄漏的问题。内存泄漏大家都不陌生了,简单粗俗的讲,...
    宇宙只有巴掌大阅读 2,329评论 0 12
  • Android 内存泄漏总结 内存管理的目的就是让我们在开发中怎么有效的避免我们的应用出现内存泄漏的问题。内存泄漏...
    apkcore阅读 1,192评论 2 7
  • 一.Java内存分配结构复习 1.Java内存分配策略 上一篇Android内存管理分析总结中我们提到了Java内...
    Geeks_Liu阅读 776评论 5 7