Handler 機制 以及 內存泄漏 筆記

主要有5個類
Handler 負責發消息和處理消息
Message 消息對象
MessageQueue 消息隊列,負責存儲消息對象 本質是優先隊列
Looper 消息輪詢器 負責從MQ中取消息並傳給Handler,讓其處理
ThreadLocal 保存Looper

發送消息

    Handler 使用 sendMessage() 方法發送消息,
    最後調用到 MessageQueue 的 enqueueMessage()方法,向隊列裏添加消息
    這裏有個關鍵點 Message 有個值 when(when = 當前時間戳+delay),添加消息的時候會將when的大小當做優先級將MessageQueue 重新排列,

輪訓消息

    Looper.loop();

接收並處理消息

    Looper 會從 MessageQueue 中取出消息
    然後會發現有這個邏輯 msg.target.dispatchMessage(msg); 其中 target 就是Handler
    Handler 的 dispatchMessage() 方法 最終調用的是 handleMessage() 方法

首先創建handler的時候必須要有Looper, 但是activity中默認已經創建了looper, 但是新開的線程,要想初始化handler,就必須調用

Looper.parepare()去創建一個,初始化handler對象之後調用, Looper.loop()處理數據. Handler跟其綁定的Looper 在同一個線程

handler的handleMessage會在dispatchMesage() 中執行, 而dispatchMessage()會在looper的loop()方法裏調用;

在loop()方法中會先獲取Looper對象

final Looper me = myLooper()

Looper會由靜態, 不可變 ThreadLocal管理, ThreadLocal 內部有個 ThreadLocalMap 來存儲對象 但是是非靜態的 所以每個線程都各有一份.

Message 從MessageQueue中取出, 而MessageQueue由looper管理, handler對象保存在Message中

總結:一個線程中可以創建多個Handler,只能存在一個Looper,只有一個message queue,通過targer去判斷哪個Handler去處理消息 

Handler造成內存泄露

當使用內部類(包括匿名類)來創建Handler的時候,Handler對象會隱式地持有一個外部類對象(通常是一個Activity)的引用(不然你怎麼可能通過Handler來操作Activity中的View?)。而Handler通常會伴隨着一個耗時的後臺線程(例如從網絡拉取圖片)一起出現,這個後臺線程在任務執行完畢(例如圖片下載完畢)之後,通過消息機制通知Handler,然後Handler把圖片更新到界面。然而,如果用戶在網絡請求過程中關閉了Activity,正常情況下,Activity不再被使用,它就有可能在GC檢查時被回收掉,但由於這時線程尚未執行完,而該線程持有Handler的引用(不然它怎麼發消息給Handler?),這個Handler又持有Activity的引用,就導致該Activity無法被回收(即內存泄露),直到網絡請求結束(例如圖片下載完畢)。另外,如果你執行了Handler的postDelayed()方法,該方法會將你的Handler裝入一個Message,並把這條Message推到MessageQueue中,那麼在你設定的delay到達之前,會有一條MessageQueue -> Message -> Handler -> Activity的鏈,導致你的Activity被持有引用而無法被回收。

方法一:通過程序邏輯來進行保護。
1.在關閉Activity的時候停掉你的後臺線程。線程停掉了,就相當於切斷了Handler和外部連接的線,Activity自然會在合適的時候被回收。 
2.如果你的Handler是被delay的Message持有了引用,那麼使用相應的Handler的removeCallbacks()方法,把消息對象從消息隊列移除就行了。

方法二:將Handler聲明爲靜態類。
PS:在Java 中,非靜態的內部類和匿名內部類都會隱式地持有其外部類的引用,靜態的內部類不會持有外部類的引用。
靜態類不持有外部類的對象,所以你的Activity可以隨意被回收。由於Handler不再持有外部類對象的引用,導致程序不允許你在Handler中操作Activity中的對象了。所以你需要在Handler中增加一個對Activity的弱引用(WeakReference)

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章