Part1_Handler內存泄漏分析及解決

一、介紹

首先,請瀏覽下面這段handler代碼:

public class SampleActivity extends Activity {

private final Handler mLeakyHandler = new Handler() {

? ? ? ? @Override

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

? ? ? ? ?// ...

? ? ? ? }

? ?}

}

在使用handler時,這是一段很常見的代碼。但是,它卻會造成嚴重的內存泄漏問題。在實際編寫中,我們往往會得到如下警告:

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

二、分析

1、 Android角度

當Android應用程序啟動時,framework會為該應用程序的主線程創建一個Looper對象。這個Looper對象包含一個簡單的消息隊列Message Queue,并且能夠循環的處理隊列中的消息。這些消息包括大多數應用程序framework事件,例如Activity生命周期方法調用、button點擊等,這些消息都會被添加到消息隊列中并被逐個處理。

另外,主線程的Looper對象會伴隨該應用程序的整個生命周期。

然后,當主線程里,實例化一個Handler對象后,它就會自動與主線程Looper的消息隊列關聯起來。所有發送到消息隊列的消息Message都會擁有一個對Handler的引用,所以當Looper來處理消息時,會據此回調[Handler#handleMessage(Message)]方法來處理消息。

2、 Java角度

在java里,非靜態內部類 和 匿名類 都會潛在的引用它們所屬的外部類。但是,靜態內部類卻不會。

三、泄露來源

請瀏覽下面一段代碼:

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();

}

}

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

<<<<<<< HEAD 注意,上面代碼中的Runnable類--非靜態匿名類--同樣持有對其外部類的引用。從而也導致泄漏。

注意,上面代碼中的Runnable類--非靜態匿名類--同樣持有對其外部類的引用。從而也導致泄漏。

四、泄漏解決方案

首先,上面已經明確了內存泄漏來源:

只要有未處理的消息,那么消息會引用handler,非靜態的handler又會引用外部類,即Activity,導致Activity無法被回收,造成泄漏;

Runnable類屬于非靜態匿名類,同樣會引用外部類。

為了解決遇到的問題,我們要明確一點:靜態內部類不會持有對外部類的引用。所以,我們可以把handler類放在單獨的類文件中,或者使用靜態內部類便可以避免泄漏。

另外,如果想要在handler內部去調用所在的外部類Activity,那么可以在handler內部使用弱引用的方式指向所在Activity,這樣統一不會導致內存泄漏。

對于匿名類Runnable,同樣可以將其設置為靜態類。因為靜態的匿名類不會持有對外部類的引用。

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();

}

}

五、小結

<<<<<<< HEAD 雖然靜態類與非靜態類之間的區別并不大,但是對于Android開發者而言卻是必須理解的。至少我們要清楚,如果一個內部類實例的生命周期比Activity更長,那么我們千萬不要使用非靜態的內部類。最好的做法是,使用靜態內部類,然后在該類里使用弱引用來指向所在的Activity。

原文鏈接:

http://www.lxweimin.com/p/cb9b4b71a820

雖然靜態類與非靜態類之間的區別并不大,但是對于Android開發者而言卻是必須理解的。至少我們要清楚,如果一個內部類實例的生命周期比Activity更長,那么我們千萬不要使用非靜態的內部類。最好的做法是,使用靜態內部類,然后在該類里使用弱引用來指向所在的Activity。

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 228,646評論 6 533
  • 序言:濱河連續發生了三起死亡事件,死亡現場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發現死者居然都...
    沈念sama閱讀 98,595評論 3 418
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 176,560評論 0 376
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 63,035評論 1 314
  • 正文 為了忘掉前任,我火速辦了婚禮,結果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 71,814評論 6 410
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發上,一...
    開封第一講書人閱讀 55,224評論 1 324
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,301評論 3 442
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 42,444評論 0 288
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當地人在樹林里發現了一具尸體,經...
    沈念sama閱讀 48,988評論 1 335
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 40,804評論 3 355
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發現自己被綠了。 大學時的朋友給我發了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 42,998評論 1 370
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,544評論 5 360
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質發生泄漏。R本人自食惡果不足惜,卻給世界環境...
    茶點故事閱讀 44,237評論 3 347
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 34,665評論 0 26
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 35,927評論 1 287
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 51,706評論 3 393
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 47,993評論 2 374

推薦閱讀更多精彩內容