Handler做爲Android代碼編寫以及面試時常常遇到的內容,有必要花個時間整理一下,畢竟寫過的東西印象會更加深入。html
源碼裏面撈出來的內容,英文不難看懂。主要就是說每一個Handler會和每一個線程以及線程對應的消息隊列相綁定。以後消息就可經過Handler在線程之間傳遞。android
A Handler allows you to send and process {@link Message} and Runnable objects associated with a thread's {@link MessageQueue}. Each Handler instance is associated with a single thread and that thread's message queue. When you create a new Handler, it is bound to the thread ,message queue of the thread that is creating it -- from that point on,
it will deliver messages and runnables to that message queue and execute them as they come out of the message queue.
<p>There are two main uses for a Handler:
(1) to schedule messages and runnables to be executed at some point in the future; and
(2) to enqueue an action to be performed on a different thread than your own.
複製代碼
Handler 的做用主要爲兩個:git
1.在將來某個時刻去執行runnable中的事件面試
handler.post(runnable,delayMillis);
複製代碼
2.不一樣消息間傳遞事件bash
android.os.Handler handler = new Handler(){
@Override
public void handleMessage(final Message msg) {
//這裏接受並處理消息
}
};
//發送消息
handler.sendMessage(message);
複製代碼
首先放一張Handler消息處理的原理圖less
首先第一步是建立Handler,Handler的構造函數主要有以下幾個 異步
其中的參數以下async
boolean asyncide
async 該參數肯定用Handler發送的消息是否要設置成異步消息,若是爲ture設置爲異步消息,異步消息可和同步阻塞一塊兒配合使用,典型的例子是界面的定時刷新。函數
Callback callback
設置了該參數,那麼Handler在dispatchMessage回調時會優先調用callback中的代碼,若返回爲true,則再也不執行handleMessage的回調
/**
* Handle system messages here.
*/
public void dispatchMessage(@NonNull Message msg) {
if (msg.callback != null) {
handleCallback(msg);
} else {
if (mCallback != null) {
if (mCallback.handleMessage(msg)) {
return;
}
}
handleMessage(msg);
}
}
複製代碼
Looper looper
設置該Hanlder對應的Looper參數,對於每一個Handler,都有一個對應的Looper,每一個Looper有其對應的MessageQueue。若是沒有顯式的設置Handler的Looper,那麼Handler默認會取該線程對應的Looper賦給該Handler。
Handler發送消息主要調用的是sendMessage方法。
sendMessage(@NonNull Message msg)
複製代碼
發送消息的方法有許多不一樣的名稱和參數。可是,你從源碼一步步看最後都會指向以下的這段代碼
public boolean sendMessageAtTime(@NonNull 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);
}
複製代碼
看這段代碼咱們會發現,其主要的操做就是調用壓隊列的方法。傳入的參數是隊列,消息以及對應的事件
該方法首先是對Message 實例進行各類參數的設置,最主要的就是target參數,該參數用於肯定消息從MessageQueue中取出後去調用哪一個Handler的dispatchMessage,從而進行消息的處理。
private boolean enqueueMessage(@NonNull MessageQueue queue, @NonNull Message msg,
long uptimeMillis) {
msg.target = this;
msg.workSourceUid = ThreadLocalWorkSource.getUid();
if (mAsynchronous) {
msg.setAsynchronous(true);
}
return queue.enqueueMessage(msg, uptimeMillis);
}
複製代碼
設置完參數,Handler的enqueueMessage就會去調用MessageQueue的enqueueMessage方法,代碼以下
boolean enqueueMessage(Message msg, long when) {
if (msg.target == null) {
throw new IllegalArgumentException("Message must have a target.");
}
if (msg.isInUse()) {
throw new IllegalStateException(msg + " This message is already in use.");
}
synchronized (this) {
if (mQuitting) {
IllegalStateException e = new IllegalStateException(
msg.target + " sending message to a Handler on a dead thread");
Log.w(TAG, e.getMessage(), e);
msg.recycle();
return false;
}
msg.markInUse();
msg.when = when;
Message p = mMessages;
boolean needWake;
if (p == null || when == 0 || when < p.when) {
// New head, wake up the event queue if blocked.
msg.next = p;
mMessages = msg;
needWake = mBlocked;
} else {
// Inserted within the middle of the queue. Usually we don't have to wake // up the event queue unless there is a barrier at the head of the queue // and the message is the earliest asynchronous message in the queue. needWake = mBlocked && p.target == null && msg.isAsynchronous(); Message prev; for (;;) { prev = p; p = p.next; if (p == null || when < p.when) { break; } if (needWake && p.isAsynchronous()) { needWake = false; } } msg.next = p; // invariant: p == prev.next prev.next = msg; } // We can assume mPtr != 0 because mQuitting is false. if (needWake) { nativeWake(mPtr); } } return true; } 複製代碼
從代碼中咱們能發現,消息隊列主要是經過鏈表的形式進行的存儲。這段代碼的所處理的事情就是根據Message中的when參數從列表中來找到須要插入的點,並把Message進行參入。
Looper取出消息主要是調用了Looper類的loop方法,該方法裏面有一個for(;;)循環,不停的從MessageQueue中取出消息。
for (;;)
{
Message msg = queue.next();
……
}
複製代碼
當消息取出之後,即會調用
msg.target.dispatchMessage(msg);
複製代碼
來處理消息,這個target即時開始enqueMessage時塞入的Handler。
從Handler中咱們能夠看到,這dispatchMessage(Message msg)方法。
public void dispatchMessage(@NonNull Message msg) {
if (msg.callback != null) {
handleCallback(msg);
} else {
if (mCallback != null) {
if (mCallback.handleMessage(msg)) {
return;
}
}
handleMessage(msg);
}
}
複製代碼
其中handleMessage(msg)就是咱們自定義Handler時的回調,這樣消息就走完了整個流程。
A: Handler的普通用法以下:
Handler handler = new Handler() {
@Override
public void handleMessage(Message msg) {
}
};
複製代碼
因爲Handler是經過匿名內部類的方式實現的,因此其會對外部類有引用(一般爲Activity),這就會引發內存泄漏。
一般的解決以下
把Handler定義成靜態內部類,如Handler中須要對外部類的參數有引用,那麼可使用弱引用,示例代碼以下
private static class MyHandler extends Handler{
//持有弱引用HandlerActivity,GC回收時會被回收掉.
private final WeakReference<HandlerActivity> mActivty;
public MyHandler(HandlerActivity activity){
mActivty =new WeakReference<HandlerActivity>(activity);
}
@Override
public void handleMessage(Message msg) {
HandlerActivity activity=mActivty.get();
super.handleMessage(msg);
if(activity!=null){
//執行業務邏輯
}
}
}
複製代碼
A: 這個的設計主要是爲了解決Handler內存泄漏的問題。
可參考以下代碼:
Handler handler = new Handler(new Handler.Callback(
@Override
public boolean handleMessage(Message msg) {
if (msg.what ==1){
textView.setText("Hello!");
return true;
}
return false;
));
複製代碼
該接口的調用主要出如今dispatchMessage中,今後處的代碼可知道,若是mCallback變量沒有定義,變量爲空,會走Handler系統的方法hadlerMessage(Message msg),而若是mCallback變量定義了,那麼其會運行到mCallback
public void dispatchMessage(@NonNull Message msg) {
if (msg.callback != null) {
handleCallback(msg);
} else {
if (mCallback != null) {
if (mCallback.handleMessage(msg)) {
return;
}
}
handleMessage(msg);
}
}
複製代碼
MessageQueue中的Message一共有三種類型,即普通消息,異步消息和消息屏障。區別是消息屏障它的target爲null。而普通消息和異步消息是經過setAsynchronous爲true來進行區分的。
在設置消息屏障後,異步消息具備優先處理的權利。界面的定時刷新就是用的這個機制
撈取Handler的源碼發現以下這段
private boolean enqueueMessage(@NonNull MessageQueue queue, @NonNull Message msg,
long uptimeMillis) {
msg.target = this;
msg.workSourceUid = ThreadLocalWorkSource.getUid();
if (mAsynchronous) {
msg.setAsynchronous(true);
}
return queue.enqueueMessage(msg, uptimeMillis);
}
複製代碼
即哪一個Handler把Message存入的消息隊列,最後回調的就是這個Handler對應的dispatchMessage 方法。
不會,若MessageQueue中消息已經取完或者消息要在以後的某個事件纔會促發,則會調用native方法 nativePollOnce(long ptr, int timeoutMillis),使主線程處於等待狀態。當調用了往消息隊列塞入消息時,則會調用nativeWake(long ptr)方法喚醒主線程,因此儘管,loop方法中有個for死循環,可是這不會致使Looper不停從消息隊列取數據。可參考Android 中 MessageQueue 的 nativePollOnce這篇文章
post方法從本質上來講也是用的Message消息傳遞的流程。
public final boolean post(@NonNull Runnable r) {
return sendMessageDelayed(getPostMessage(r), 0);
}
複製代碼
private static Message getPostMessage(Runnable r) {
Message m = Message.obtain();
m.callback = r;
return m;
}
複製代碼
從以下兩端代碼可看出其是對runnable進行了封裝,把runnable塞進了Message中。而當Handler得到消息後,其會取出msg.callback參數,而後運行裏面的runnable方法。
用Message.obtain()獲取比較好,避免了gc致使的性能消耗。
IdleHandler是在主線程消息隊列空閒時,會被取出執行的對象。可在一些頁面優化的情景下使用。 可參考我以前寫的一篇文章 IdleHandler,頁面啓動優化神器