Android Handler内存泄漏分析与解决方法

Android的UI也是线程安全的,开发者无法在子线程中更新UI,而必须在主线程中进行,所以开发者需要用到Android提供的异步消息处理机制

为什么Handler会造成内存泄漏

下面是一段简单的Handler使用

private Handler myHandler = new Handler(){
        @Override
        public void handleMessage(Message msg) {
            super.handleMessage(msg);
        }
    };

在Java中,非静态内部类会隐式的持有一个外部类对象的引用
当使用内部类(匿名类)来创建Handler的时候,Handler会隐式的持有一个外部类对象(Activity)的引用。一般开发者在子线程中处理一些耗时的任务的时候都会使用到Handler,当该任务还未完成时,用户就finish()掉这个Activity。正常情况下该Activity不再使用了,则GC(垃圾回收机制)检查的时候会把它回收掉。但是由于子线程中的任务尚未完成,子线程中的Message会持有一个Handler的引用,而Handler又持有这Activity的引用,联系如下:


引用图

所以这就导致Activity无法被回收,引发内存泄漏的问题

怎么解决

在Android Studio中使用上面的方式创建Handler的时候会有警告,内容是:This Handler class should be static or leaks might occur
由提示信息可知,只需要把Handler声明成静态内部类就行了,静态内部类不持有外部类对象的引用,Activity就不会因为Handler的原因而无法被回收了

private static class MyHandler extends Handler{

        @Override
        public void handleMessage(Message msg) {
            super.handleMessage(msg);
        }
    }

上面创建Handler的方式看似已经把问题解决了,但是Handler不再持有Activity的引用了,所以导致程序不允许在Handler中操作Activity中的对象了,这时就需要在Handler中增加一个对Activity的弱引用(WeakReference)

    private static class MyHandler extends Handler{

        WeakReference<MainActivity> mainActivity;

        public MyHandler(MainActivity activity){
            mainActivity = new WeakReference<>(activity);
        }

        @Override
        public void handleMessage(Message msg) {
            if (mainActivity.get() != null){
                
            }
        }
    }

下面补充一点引用的知识:

引用类型 说明
强引用 当内存空 间不足,Java虚拟机宁愿抛出OutOfMemoryError错误,使程序异常终止,也不会靠随意回收具有强引用的对象来解决内存不足问题
软引用 (SoftReference) 在内存不足时,GC会回收软引用指向的对象
弱引用(WeakReference) 不管内存足不足,GC都可能回收弱引用指向的对象
虚引用(PhantomReference ) 如果一个对象仅持有虚引用,那么它就和没有任何引用一样,在任何时候都可能被垃圾回收

除了通过弱应用的方式来解决Handler内存泄漏的问题之外,还有一种解决办法,就是在Activity关闭的时候使用相对应的removeCallbacks()方法来把消息对象从消息队列中删除,例如:

public class MainActivity extends AppCompatActivity {

    private  Handler myHandler = new Handler() {
            @Override
            public void handleMessage(Message msg) {
                super.handleMessage(msg);

            }
    };

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
    }

    @Override
    protected void onDestroy() {
        super.onDestroy();
        myHandler.removeCallbacksAndMessages(null);
    }
}

总结

内存泄露在 Android 开发中是一个比较严重的问题,系统给每一个应用分配的内存是固定的,一旦发生了内存泄露,就会导致该应用可用内存越来越小,严重时会发生 OOM 导致 Force Close,所以在平时的开发中应该避免引发内存泄漏

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

推荐阅读更多精彩内容

  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 172,012评论 25 707
  • Android 内存管理的目的 内存管理的目的就是让我们在开发中怎么有效的避免我们的应用出现内存泄漏的问题。简单粗...
    晨光光阅读 1,291评论 1 4
  • 信息断舍离 “订阅很多公众号,却满屏都是未读的小红点。” 互联网信息大爆炸,垃圾信息特别多,在这当中游行不撞船,要...
    陈兆丰丨混沌广州阅读 345评论 0 0
  • 没有方向的奋力奔跑,终是南辕北辙的结局 没有思考的生活,终是辨不清方向的航行 每一个生命的降临,都在演绎着一场小丑...
    张娜29阅读 316评论 0 2
  • 从小,我就表现得与众不同。还是婴儿时期的我不尿裤子不哭闹;六个月会说话;一岁就能四维清晰的说好多与年龄不同的话,不...
    霉girl阅读 287评论 1 1