一卷扮、介紹
首先,請(qǐng)瀏覽下面這段handler代碼:
public class SampleActivity extends Activity {
private final Handler mLeakyHandler = new Handler() {
@Override
public void handleMessage(Message msg) {
// ...
}
}
}
在使用handler時(shí)谨读,這是一段很常見的代碼局装。但是,它卻會(huì)造成嚴(yán)重的內(nèi)存泄漏問題劳殖。在實(shí)際編寫中铐尚,我們往往會(huì)得到如下警告:
In Android, Handler classes should be static or leaks might occur.
那么,handler是如何造成內(nèi)存泄漏的呢哆姻?
二宣增、分析
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)起來肋拔。所有發(fā)送到消息隊(duì)列的消息Message都會(huì)擁有一個(gè)對(duì)Handler的引用,所以當(dāng)Looper來處理消息時(shí)呀酸,會(huì)據(jù)此回調(diào) Handler.handleMessage()方法來處理消息
2凉蜂、 Java角度
在java里,非靜態(tài)內(nèi)部類(包括匿名類)都會(huì)潛在的引用它們所屬的外部類性誉。但是窿吩,靜態(tài)內(nèi)部類卻不會(huì)。
三错览、泄漏來源
請(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)用程序的泄漏申尤。
注意癌幕,上面代碼中的Runnable類--非靜態(tài)匿名類--同樣持有對(duì)其外部類的引用。從而也導(dǎo)致泄漏昧穿。
四勺远、泄漏解決方案
首先,上面已經(jīng)明確了內(nèi)存泄漏來源:
- 只要有未處理的消息时鸵,那么消息會(huì)引用handler胶逢,非靜態(tài)的handler又會(huì)引用外部類,即Activity饰潜,導(dǎo)致Activity無法被回收初坠,造成泄漏;
- Runnable類屬于非靜態(tài)匿名類彭雾,同樣會(huì)引用外部類碟刺。
為了解決遇到的問題,我們要明確一點(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<SampleActivity> mActivity;
public MyHandler(SampleActivity activity) {
mActivity = new WeakReference<SampleActivity>(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é)
雖然靜態(tài)類與非靜態(tài)類之間的區(qū)別并不大,但是對(duì)于Android開發(fā)者而言卻是必須理解的谬哀。至少我們要清楚刺覆,如果一個(gè)內(nèi)部類實(shí)例的生命周期比Activity更長(zhǎng),那么我們千萬不要使用非靜態(tài)的內(nèi)部類史煎。最好的做法是谦屑,使用靜態(tài)內(nèi)部類,然后在該類里使用弱引用來指向所在的Activity篇梭。
文章為轉(zhuǎn)載氢橙,原文