其实网络上已经有很多关于handler的内存泄漏的讲解和处理了,此处作为自己的笔记记录下.
先来一段比较常见的代码
package com.jetair.ordercenter.ui.activity;
import android.os.Handler;
import android.os.Message;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import com.jetair.ordercenter.R;
public class TestActivity extends AppCompatActivity {
private Handler handler;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_test);
handler = new Handler(){
@Override
public void handleMessage(Message msg) {
super.handleMessage(msg);
}
};
Message message = new Message();
message.what = 1001;
handler.sendMessageDelayed(message, 1000);
}
}
这段代码很常见也很简单,就是创建一个handler然后发送一个延时消息,这里的延时消息模拟一个耗时的操作,但是这段代码就已经会出现内存泄漏了,其实编译器已经提醒你了,将鼠标放到handler上,就会出现如下提示(handler应该是静态的)
当然,使用lint或者leakcanary的话也是会提醒有内存泄漏的.
handler内存泄漏的原理
在java中非静态内部类(包括匿名内部类)创建的时候会持有一个外部类的引用,也就是handler创建的时候会持有一个activity的引用,当消息发送到与Looper相关联的消息队列的时候,此时就会持有一个handler的引用,这样就等于一直持有一个activity的引用,当GC发生时,activity无法被回收,因此会产生内存泄漏
解决方法
方法一: 当然activity销毁的时候清空消息队列
@Override
protected void onDestroy() {
super.onDestroy();
if (handler != null){
handler.removeCallbacksAndMessages(null);
}
}
方法二:采用static+WeakReference(弱引用)的方式
package com.jetair.ordercenter.ui.activity;
import android.app.Activity;
import android.os.Handler;
import android.os.Message;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.widget.Button;
import com.jetair.ordercenter.R;
import java.lang.ref.WeakReference;
public class TestActivity extends AppCompatActivity {
private MyHandler handler;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_test);
handler = new MyHandler(this);
Message message = new Message();
message.what = 1001;
handler.sendMessageDelayed(message, 1000);
}
static class MyHandler extends Handler {
WeakReference<Activity> mActivityReference;
public MyHandler(Activity activity) {
mActivityReference = new WeakReference<Activity>(activity);
}
@Override
public void handleMessage(Message msg) {
TestActivity activity = (TestActivity) mActivityReference.get();
}
}
@Override
protected void onDestroy() {
super.onDestroy();
if (handler != null){
handler.removeCallbacksAndMessages(null);
}
}
}