Android消息機制Handler用法總結

1.簡述

Handler消息機制主要包括: MessageQueue、 Handler、 Looper、Message。

  • Message:需要傳遞的消息,可以傳遞數據;
  • MessageQueue:消息隊列,但是它的內部實現並不是用的隊列,而是通過單鏈表的數據結構來維護消息列表,因為單鏈表在插入和刪除上比較有優勢。主要功能是向消息池投遞消息( MessageQueue.enqueueMessage)和取走消息池的消息( MessageQueue.next)。 
  • Handler:消息輔助類,主要功能是向消息池發送各種消息事件( Handler.sendMessage)和處理相應消息事件( Handler.handleMessage); 
  • Looper:消息控制器,不斷循環執行( Looper.loop),從MessageQueue中讀取消息,按分發機制將消息分發給目標處理者。

2.異步線程切換原理

class LooperThread extends Thread {
      public Handler mHandler;
 
      public void run() {
          Looper.prepare();
 
          mHandler = new Handler() {
              public void handleMessage(Message msg) {
                  // process incoming messages here
              }
          };
 
          Looper.loop();
      }
  }

每一個異步線程,都維護著唯一的一個Looper,每一個Looper會初始化(維護)一個MessageQueue,之後進入一個無限循環一直在讀取MessageQueue中存儲的消息,如果沒有消息那就一直阻塞等待。

我們在實例化handler的過程中,會先得到當前所在線程的looper對象,之後得到與該looper對象相對應的消息隊列,看源碼Handler中持有Looper、MessageQueue。

    private static void handleCallback(Message message) {
        message.callback.run();
    }
 
    final Looper mLooper;
    final MessageQueue mQueue;
    final Callback mCallback;

當我們發送消息的時候,即handler.sendMessage或者handler.post,會將msg中的target賦值為handler自身,這就是實現message從一個線程到另外一個線程的傳遞的本質,之後加入到消息隊列中。

private boolean enqueueMessage(MessageQueue queue, Message msg, long uptimeMillis) {
    msg.target = this;
    if (mAsynchronous) {
        msg.setAsynchronous(true);
    }
    return queue.enqueueMessage(msg, uptimeMillis);
}

我們一般會重寫handlerMessage方法處理消息,這將會在msg.target.disPatchMessage方法中被回調,從而實現瞭message從一個線程到另外一個線程的傳遞。

3.總結

  • 1.Handler 的背後有 Looper、MessageQueue 支撐,Looper 負責消息分發,MessageQueue 負責消息管理;
  • 2.在創建 Handler 之前一定需要先創建 Looper,Looper 有退出的功能,但是主線程的 Looper 不允許退出;
  • 3.異步線程Looper,Looper.prepare()創建Looper,Looper.loop()開始輪詢,需要自己調用 Looper.myLooper().quit()退出;
  • 4.Runnable 被封裝進瞭 Message,可以說是一個特殊的 Message;
  • 5.Handler.handleMessage() 所在的線程是 Looper.loop() 方法被調用的線程;
  • 6.Handler內存泄漏的原因

原因:MessageQueue持有Message,Message持有activity,delay多久,message就會持有activity多久。

解決方案:靜態內部類、弱引用,最後不要忘記調用Handler.removeCallbacksAndMessages(null)清空所有消息。

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) {
        // ...
      }
    }
  }
 
 // MyHandler
 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();
  }
 
  @Override
  protected void onDestroy() {
        super.onDestroy();
        //mHandler.removeCallbacksAndMessages(null);
    }
}

以上所述是小編給大傢介紹的Android消息機制Handler用法總結,希望對大傢有所幫助。在此也非常感謝大傢對WalkonNet網站的支持!

推薦閱讀: