天天看點

android中handler記憶體洩漏解決方法,Handler記憶體洩漏原因及解決方案

Handler記憶體洩漏原因及解決方案

目錄:

android中handler記憶體洩漏解決方法,Handler記憶體洩漏原因及解決方案

1.須知:

主線程Looper生命周期和Activity的生命周期一緻。

非靜态内部類,或者匿名内部類。預設持有外部類引用。

2.原因:

Handler造成記憶體洩露的原因。非靜态内部類,或者匿名内部類。使得Handler預設持有外部類的引用。在Activity銷毀時,由于Handler可能有未執行完/正在執行的Message。導緻Handler持有Activity的引用。進而導緻GC無法回收Activity。

3.可能造成記憶體洩漏

匿名内部類:

//匿名内部類

Handler handler=new Handler(){

@Override

public void handleMessage(Message msg) {

super.handleMessage(msg);

}

};

複制代碼非靜态内部類:

//非靜态内部類

protected class AppHandler extends Handler {

@Override

public void handleMessage(Message msg) {

switch (msg.what) {

// TODO: 2019/4/30

}

}

}

複制代碼

4.解決方法:

Activity銷毀時,清空Handler中,未執行或正在執行的Callback以及Message。

// 清空消息隊列,移除對外部類的引用

@Override

protected void onDestroy() {

super.onDestroy();

mHandler.removeCallbacksAndMessages(null);

}

//Handler源碼中removeCallbacksAndMessages()注釋含義

public final void removeCallbacksAndMessages(Object token) {

mQueue.removeCallbacksAndMessages(this, token);

}

複制代碼靜态内部類+弱引用

private static class AppHandler extends Handler {

//弱引用,在垃圾回收時,被回收

WeakReference activity;

AppHandler(Activity activity){

this.activity=new WeakReference(activity);

}

public void handleMessage(Message message){

switch (message.what){

//todo

}

}

}

複制代碼

5. 其他:

即使記憶體洩漏了。待handler中的消息處理完。下次GC時即可回收本次未回收的記憶體。