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í)谨读,這是一段很常見的代碼局装。但是,它卻會(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)載氢橙,原文

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市恬偷,隨后出現(xiàn)的幾起案子悍手,更是在濱河造成了極大的恐慌,老刑警劉巖袍患,帶你破解...
    沈念sama閱讀 216,591評(píng)論 6 501
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件坦康,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡诡延,警方通過查閱死者的電腦和手機(jī)滞欠,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,448評(píng)論 3 392
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來肆良,“玉大人筛璧,你說我怎么就攤上這事∪鞘眩” “怎么了夭谤?”我有些...
    開封第一講書人閱讀 162,823評(píng)論 0 353
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)座舍。 經(jīng)常有香客問我沮翔,道長(zhǎng)陨帆,這世上最難降的妖魔是什么曲秉? 我笑而不...
    開封第一講書人閱讀 58,204評(píng)論 1 292
  • 正文 為了忘掉前任采蚀,我火速辦了婚禮,結(jié)果婚禮上承二,老公的妹妹穿的比我還像新娘榆鼠。我一直安慰自己,他們只是感情好亥鸠,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,228評(píng)論 6 388
  • 文/花漫 我一把揭開白布妆够。 她就那樣靜靜地躺著,像睡著了一般负蚊。 火紅的嫁衣襯著肌膚如雪神妹。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 51,190評(píng)論 1 299
  • 那天家妆,我揣著相機(jī)與錄音鸵荠,去河邊找鬼。 笑死伤极,一個(gè)胖子當(dāng)著我的面吹牛蛹找,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播哨坪,決...
    沈念sama閱讀 40,078評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼庸疾,長(zhǎng)吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來了当编?” 一聲冷哼從身側(cè)響起届慈,我...
    開封第一講書人閱讀 38,923評(píng)論 0 274
  • 序言:老撾萬榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎凌箕,沒想到半個(gè)月后拧篮,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,334評(píng)論 1 310
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡牵舱,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,550評(píng)論 2 333
  • 正文 我和宋清朗相戀三年串绩,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片芜壁。...
    茶點(diǎn)故事閱讀 39,727評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡礁凡,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出慧妄,到底是詐尸還是另有隱情顷牌,我是刑警寧澤,帶...
    沈念sama閱讀 35,428評(píng)論 5 343
  • 正文 年R本政府宣布塞淹,位于F島的核電站窟蓝,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏饱普。R本人自食惡果不足惜运挫,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,022評(píng)論 3 326
  • 文/蒙蒙 一状共、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧谁帕,春花似錦峡继、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,672評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至儡循,卻和暖如春舶吗,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背择膝。 一陣腳步聲響...
    開封第一講書人閱讀 32,826評(píng)論 1 269
  • 我被黑心中介騙來泰國(guó)打工裤翩, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人调榄。 一個(gè)月前我還...
    沈念sama閱讀 47,734評(píng)論 2 368
  • 正文 我出身青樓踊赠,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親每庆。 傳聞我的和親對(duì)象是個(gè)殘疾皇子筐带,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,619評(píng)論 2 354

推薦閱讀更多精彩內(nèi)容