在Activity或者Fragment中创建异步处理对象时,比如Handler、AsyncTask,要特别注意内存泄漏问题,这些异步对象在子线程处理请求,生命周期和外面Activity和Fragment存在不同步的问题,当Activity的生命周期结束时,若不显式在onDestroy中将这些异步控件结束,将会造成activity无法被回收,也就造成了内存泄漏。
下面我们从smali文件和内存堆栈两个角度分析内存泄漏问题。
public class HandlerTestActivity extends Activity {
private static final int MESSAGE_WHAT = 1;
@Override
protected void onCreate(@Nullable Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
mHandler.sendMessageDelayed(mHandler.obtainMessage(MESSAGE_WHAT), 50000);
startActivity(new Intent(this, TestActivity.class));
finish();
}
private Handler mHandler = new Handler(){
@Override
public void handleMessage(Message msg) {
super.handleMessage(msg);
switch (msg.what) {
case MESSAGE_WHAT:
doSomething();//handler调用外部类的方法,将会持有外部类的隐式调用
break;
}
}
};
private void doSomething(){
Log.i("", "doSomething");
}
}
测试类
# virtual methods
.method public handleMessage(Landroid/os/Message;)V
.locals 1
.param p1, "msg" # Landroid/os/Message;
.prologue
.line 38
invoke-super {p0, p1}, Landroid/os/Handler;->handleMessage(Landroid/os/Message;)V
.line 40
iget-object v0, p0, Lcom/jason/handlertest/HandlerTestActivity$1;->this$0:Lcom/jason/handlertest/HandlerTestActivity;
# invokes: Lcom/jason/handlertest/HandlerTestActivity;->doSomething()V
invoke-static {v0}, Lcom/jason/handlertest/HandlerTestActivity;->access$000(Lcom/jason/handlertest/HandlerTestActivity;)V
.line 43
return-void
.end method
handler延迟50s发送一条message,在此期间handleMessage方法中将会持有外部HandlerTestActivity的引用,50s内activity将无法释放,如果handler处理的是循环消息,那么activity将不会释放
下面我们看一下内存情况
首次启动的内存
多次启动后,手动gc后的内存情况
怎么样避免这种情况?
@Override
protected void onDestroy() {
super.onDestroy();
mHandler.removeMessages(MESSAGE_WHAT);
}
1.在activity结束时,清空handler的消息并退出
class MyHandler extends Handler{
WeakReference<Activity> activityWeakReference;
public MyHandler(Activity activity){
activityWeakReference = new WeakReference<>(activity);
}
@Override
public void handleMessage(Message msg) {
super.handleMessage(msg);
switch (msg.what){
case MESSAGE_WHAT:
if (activityWeakReference.get() != null){
((HandlerTestActivity)activityWeakReference.get()).doSomething();
}
break;
}
}
}
2.内部类Handler采用弱引用持有activity
private static Handler mHandler = new Handler(){
@Override
public void handleMessage(Message msg) {
super.handleMessage(msg);
switch (msg.what) {
case MESSAGE_WHAT:
doSomething();//handler调用外部类的方法,将会持有外部类的隐式调用
break;
}
}
};
3.handler采用静态,这种方式缺点是调用的方法也变为了静态
4.采用RxJava CompositeSubscription订阅的方式,在activity退出时调用CompositeSubscription.unsubscribe()
参考文章
http://www.jianshu.com/p/fa3b31f8f47d