Part1_Handler內(nèi)存泄漏分析及解決

一、介紹

首先,請(qǐng)瀏覽下面這段handler代碼:

public class SampleActivity extends Activity {

private final Handler mLeakyHandler = new Handler() {

? ? ? ? @Override

? ? ? ? ?public void handleMessage(Message msg) {

? ? ? ? ?// ...

? ? ? ? }

? ?}

}

在使用handler時(shí),這是一段很常見(jiàn)的代碼。但是,它卻會(huì)造成嚴(yán)重的內(nèi)存泄漏問(wèn)題。在實(shí)際編寫中,我們往往會(huì)得到如下警告:

? In Android, Handler classes should be static or leaks might occur.

二、分析

1、 Android角度

當(dāng)Android應(yīng)用程序啟動(dòng)時(shí),framework會(huì)為該應(yīng)用程序的主線程創(chuàng)建一個(gè)Looper對(duì)象。這個(gè)Looper對(duì)象包含一個(gè)簡(jiǎn)單的消息隊(duì)列Message Queue,并且能夠循環(huán)的處理隊(duì)列中的消息。這些消息包括大多數(shù)應(yīng)用程序framework事件,例如Activity生命周期方法調(diào)用、button點(diǎn)擊等,這些消息都會(huì)被添加到消息隊(duì)列中并被逐個(gè)處理。

另外,主線程的Looper對(duì)象會(huì)伴隨該應(yīng)用程序的整個(gè)生命周期。

然后,當(dāng)主線程里,實(shí)例化一個(gè)Handler對(duì)象后,它就會(huì)自動(dòng)與主線程Looper的消息隊(duì)列關(guān)聯(lián)起來(lái)。所有發(fā)送到消息隊(duì)列的消息Message都會(huì)擁有一個(gè)對(duì)Handler的引用,所以當(dāng)Looper來(lái)處理消息時(shí),會(huì)據(jù)此回調(diào)[Handler#handleMessage(Message)]方法來(lái)處理消息。

2、 Java角度

在java里,非靜態(tài)內(nèi)部類 和 匿名類 都會(huì)潛在的引用它們所屬的外部類。但是,靜態(tài)內(nèi)部類卻不會(huì)。

三、泄露來(lái)源

請(qǐng)瀏覽下面一段代碼:

public class SampleActivity extends Activity {

private final Handler mLeakyHandler = new Handler() {

@Override

public void handleMessage(Message msg) {

// ...

}

}

@Override

protected void onCreate(Bundle savedInstanceState) {

super.onCreate(savedInstanceState);

// Post a message and delay its execution for 10 minutes.

mLeakyHandler.postDelayed(new Runnable() {

@Override

public void run() { /* ... */ }

}, 1000 * 60 * 10);

// Go back to the previous Activity.

finish();

}

}

當(dāng)activity結(jié)束(finish)時(shí),里面的延時(shí)消息在得到處理前,會(huì)一直保存在主線程的消息隊(duì)列里持續(xù)10分鐘。而且,由上文可知,這條消息持有對(duì)handler的引用,而handler又持有對(duì)其外部類(在這里,即SampleActivity)的潛在引用。這條引用關(guān)系會(huì)一直保持直到消息得到處理,從而,這阻止了SampleActivity被垃圾回收器回收,同時(shí)造成應(yīng)用程序的泄漏。

<<<<<<< HEAD 注意,上面代碼中的Runnable類--非靜態(tài)匿名類--同樣持有對(duì)其外部類的引用。從而也導(dǎo)致泄漏。

注意,上面代碼中的Runnable類--非靜態(tài)匿名類--同樣持有對(duì)其外部類的引用。從而也導(dǎo)致泄漏。

四、泄漏解決方案

首先,上面已經(jīng)明確了內(nèi)存泄漏來(lái)源:

只要有未處理的消息,那么消息會(huì)引用handler,非靜態(tài)的handler又會(huì)引用外部類,即Activity,導(dǎo)致Activity無(wú)法被回收,造成泄漏;

Runnable類屬于非靜態(tài)匿名類,同樣會(huì)引用外部類。

為了解決遇到的問(wèn)題,我們要明確一點(diǎn):靜態(tài)內(nèi)部類不會(huì)持有對(duì)外部類的引用。所以,我們可以把handler類放在單獨(dú)的類文件中,或者使用靜態(tài)內(nèi)部類便可以避免泄漏。

另外,如果想要在handler內(nèi)部去調(diào)用所在的外部類Activity,那么可以在handler內(nèi)部使用弱引用的方式指向所在Activity,這樣統(tǒng)一不會(huì)導(dǎo)致內(nèi)存泄漏。

對(duì)于匿名類Runnable,同樣可以將其設(shè)置為靜態(tài)類。因?yàn)殪o態(tài)的匿名類不會(huì)持有對(duì)外部類的引用。

public class SampleActivity extends Activity {

/**

* Instances of static inner classes do not hold an implicit

* reference to their outer class.

*/

private static class MyHandler extends Handler {

private final WeakReference mActivity;

public MyHandler(SampleActivity activity) {

mActivity = new WeakReference(activity);

}

@Override

public void handleMessage(Message msg) {

SampleActivity activity = mActivity.get();

if (activity != null) {

// ...

}

}

}

private final MyHandler mHandler = new MyHandler(this);

/**

* Instances of anonymous classes do not hold an implicit

* reference to their outer class when they are "static".

*/

private static final Runnable sRunnable = new Runnable() {

@Override

public void run() { /* ... */ }

};

@Override

protected void onCreate(Bundle savedInstanceState) {

super.onCreate(savedInstanceState);

// Post a message and delay its execution for 10 minutes.

mHandler.postDelayed(sRunnable, 1000 * 60 * 10);

// Go back to the previous Activity.

finish();

}

}

五、小結(jié)

<<<<<<< HEAD 雖然靜態(tài)類與非靜態(tài)類之間的區(qū)別并不大,但是對(duì)于Android開(kāi)發(fā)者而言卻是必須理解的。至少我們要清楚,如果一個(gè)內(nèi)部類實(shí)例的生命周期比Activity更長(zhǎng),那么我們千萬(wàn)不要使用非靜態(tài)的內(nèi)部類。最好的做法是,使用靜態(tài)內(nèi)部類,然后在該類里使用弱引用來(lái)指向所在的Activity。

原文鏈接:

http://www.itdecent.cn/p/cb9b4b71a820

雖然靜態(tài)類與非靜態(tài)類之間的區(qū)別并不大,但是對(duì)于Android開(kāi)發(fā)者而言卻是必須理解的。至少我們要清楚,如果一個(gè)內(nèi)部類實(shí)例的生命周期比Activity更長(zhǎng),那么我們千萬(wàn)不要使用非靜態(tài)的內(nèi)部類。最好的做法是,使用靜態(tài)內(nèi)部類,然后在該類里使用弱引用來(lái)指向所在的Activity。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時(shí)請(qǐng)結(jié)合常識(shí)與多方信息審慎甄別。
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡(jiǎn)書系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

友情鏈接更多精彩內(nèi)容