【Android内存优化】内存泄露优化之强引用变弱引用完全详解

本文深入探讨了Android内存泄露问题,解析了内存泄露的定义和JVM垃圾回收原理。通过分析Handler引起的内存泄露案例,指出非静态内部类持有的Activity强引用可能导致内存无法释放。建议使用静态内部类结合WeakReference创建弱引用Handler,以避免内存泄露,并提供了解决方案,包括在退出时清除Handler的回调和消息。
摘要由CSDN通过智能技术生成

内存泄露背景

什么是内存泄露

内存空间使用完毕后无法被释放的现象,对于还保持着引用, 该内存不能再被分配使用,逻辑上却已经不会再用到的对象,垃圾回收器不会回收它们。
所以逻辑不再使用的对象,需要释放强引用,以便GC进行回收。
在这里插入图片描述

JVM 工作原理

JVM 垃圾回收原理,点击进入

内存泄露案例

  • 常见Handler 写法
	Handler mHandler = new Handler(){
            @Override
            public void handleMessage(Message msg) {
                super.handleMessage(msg);
                switch (msg.what) {
                    case 0:
                        // do something
                        sendEmptyMessageDelayed(100, 2000);
                        break;
                    case 1:
                        // do something
                        break;
                    default:break;
                }
            }
        };
	mHandler.sendEmptyMessage(1);

This Handler class should be static or leaks might occur (anonymous android.os.Handler)

  • Handler 内存泄漏说明

mHandler 会作为成员变量保存在发送的消息msg 中,即 msg 持有mHandler 的引用,而 mHandler 是 Activity 的非静态内部类实例,即mHandler 持有 Activity 的引用,那么我们就可以理解为 msg 间接持有 Activity 的引用。msg 被发送后先放到消息队列 MessageQueue 中,然后等待 Looper 的轮询处理(MessageQueue 和 Looper 都是与线程相关联的, MessageQueue 是 Looper 引用的成员变量,而 Looper 是保存在ThreadLocal 中的)。那么当 Activity 退出后,msg 可能仍然存在于消息对列MessageQueue 中未处理或者正在处理,那么这样就会导致 Activity 无法被回收,以致发生Activity 的内存泄露。

通常在Android 开发中如果要使用内部类,但又要规避内存泄露,一般都会采用 静态内部类 + 弱引用 的方式。

内存泄露优化

  1. 自定义一个静态内部类 Handler

用静态内部类的方式自定义一个 Handler ,使用 WeakReference 实现弱引用。
传入的 SecondActivity 是使用这个 Handler 的 Activity ,按照实际情况传入即可; 传入Fragment 也是可以的。

private static class SecondHandler extends Handler{

        private WeakReference<SecondActivity> mWeakReference;

        public SecondHandler(SecondActivity secondActivity) {
            mWeakReference = new WeakReference<>(secondActivity);
        }

        @Override
        public void handleMessage(Message msg) {
            SecondActivity host = mWeakReference.get(); // 判断所在的 Activity 的引用是否被回收了
            if (host != null) {
                switch (msg.what){
                    case 99:
                        // do something
                        sendEmptyMessageDelayed(100, 2000);
                        break;
                    case 100:
                        // do something
                        break;
                    default:break;
                }
            }
        }
    }

mHandler 通过弱引用的方式持有Activity,当GC 执行垃圾回收时,遇到Activity 就会回收并释放所占据的内存单元。这样就不会发生内存泄露了。

  1. 弱引用 Handler 初始化
    参考构造函数,初始化也很简单,使用的话还是 Handler 的用法。
SecondHandler mHandler = new SecondHandler(SecondActivity.this);
mHandler.sendEmptyMessage(1);
  1. 退出时移除回调和消息
    当 Activity 退出后,将 mHandler 的回调和消息移除。在 onStop 或者 onDestroy 中执行即可。
@Override
protected void onStop() {
	super.onStop();
	mHandler.removeCallbacksAndMessages(null);
}

传入 null 是移除所有回调和消息,查看 Handler 源码可知,


/**
 * Remove any pending posts of callbacks and sent messages whose
 * <var>obj</var> is <var>token</var>.  If <var>token</var> is null,
 * all callbacks and messages will be removed.
 *   */
public final void removeCallbacksAndMessages(Object token) {
	mQueue.removeCallbacksAndMessages(this, token);
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值