其實問這種問題,無非就是想知道你對handler瞭解怎麼樣。雖然handler這種問題已經被問爛了,可是不少公司,尤爲是中小型企業,面試
被問的概率仍是很高,因此,再次記錄一下,也很簡單,沒事兒的時候點進源碼看看就明白。多線程
Android 的 多線程通信核心類是Handler!async
一般咱們在使用多線程通信的方式,是在 線程A(或UI線程) 中建立 handler,在線程B中 使用 handler的實例,調用其方法 例如:
sendMessage(Message msg)
、sendEmptyMessage(int what)
,發送咱們須要傳送的數據放進了Messagequeue裏面。而後被主線程或者是實例化化handler的線程 的Looper循環到,而後執行。ide
經過源碼咱們大體能夠理出一條線路來:oop
須要注意的是,
ThreadLocal
該類是用於建立線程局部變量的類,咱們知道,一般狀況下一個變量是在多個線程中都能訪問的,可是ThreadLocal,可建立一個只在當前線程訪問的變量。他支持泛型,回到咱們的面試題上ui
這裏 咱們能夠逐步分析:this
public Handler(Callback callback, boolean async) { //獲得主線程(已經被ActivityThread初始化好)的looper mLooper = Looper.myLooper(); if (mLooper == null) { throw new RuntimeException( "Can't create handler inside thread that has not called Looper.prepare()"); } //獲得主線程(已經被ActivityThread初始化好)的looper的MessageQueue,註釋:① mQueue = mLooper.mQueue; mCallback = callback; mAsynchronous = async; }
注意,看到了嗎? 這裏是核心:
mLooper = Looper.myLooper();
,spa
public static Looper myLooper() { return sThreadLocal.get(); }
而這個Looper對象,是在咱們啓動程序的時候,也就是ActivityThread 中的main方法幫咱們初始化的,也就是咱們主線程的Looper。線程
public static void prepareMainLooper() { ··· prepare(false); ··· } -------------------------------------------------- private static void prepare(boolean quitAllowed) { if (sThreadLocal.get() != null) { throw new RuntimeException("Only one Looper may be created per thread"); } sThreadLocal.set(new Looper(quitAllowed)); }
咱們在子線程 使用 handler的實例進行 sendMessage(Message msg)
的時候:code
public boolean sendMessageAtTime(Message msg, long uptimeMillis) { //這裏的代碼和上面的註釋①對應。 MessageQueue queue = mQueue; if (queue == null) { RuntimeException e = new RuntimeException( this + " sendMessageAtTime() called with no mQueue"); Log.w("Looper", e.getMessage(), e); return false; } return enqueueMessage(queue, msg, uptimeMillis); }
最終走向enqueueMessage
:
private boolean enqueueMessage(MessageQueue queue, Message msg, long uptimeMillis) { msg.target = this; if (mAsynchronous) { msg.setAsynchronous(true); } return queue.enqueueMessage(msg, uptimeMillis); }
最終把咱們的Message,放進了主線程的MessageQueue裏面進行循環!