但是他是預設在Main Thread, 如果要多開Thread來使用Hanlder,
則必須自行控制Looper,
相對上比較麻煩, 因此Android開發了HandlerThread讓你能夠方便操作。
HandlerThread整合了Thread、Looper和MessageQueue。
啟動方式跟Thread相同。
HandlerThread handlerThread = new HandlerThread("HandlerThread");
handlerThread.start();
Handler mHandler = new Hanlder(handlerThread.getLooper()){
public void handleMessage(Message msg){
super.handleMessage(msg);
switch(msg.what){
case 1:
Logger.e("message receive");
break;
}
}
}
handler.sendEmptyMessage(1);
//或者
new Hanlder(handlerThread.getLooper()).post(new Runnable(){
public void run(){
//長時間任務1
//長時間任務2
}
});
上面的例子一旦執行, 除非你手動停止, 否則該thread會一直等待執行下去。 利用Handler丟Message,
透過Looper循序message發送, 因此確保是執行緒安全的,
但是相對缺點就是效率會下降。
HandlerThread也有生命周期
- Creation(建立):
HandlerThread建構子有名稱, 或者名稱跟優先權
HandlerThread(String name)
HandlerThread(String name, int priority)
- Execution(執行):
代表已經被start了, 隨時可以接受訊息到來。 - Reset(重設):
你可以將Queue內的訊息清空, 除了已經送出去的Message或者正在執行的Message
mHandlerThread.removeCallbackAndMessages(null);
- Termination(終止):
public void stopHandlerThread(HandlerThread handlerThread){
handlerThread.quit();
handlerThread.interrupt();
}
//或者直接在handler上面
handler.post(new Runnable(){
public void run(){
Looper.myLooper.quit();
}
});
HandlerThread確保是循序且執行緒安全的, 有人會覺得既然如此, Thread也可以達成這樣的需求, 是沒錯, 不過這樣變成你必須在程式碼內同時寫在同一個Thread內。
new Thread(new Runnable(){
public void run(){
//task 1
//task 2
//task 3
}
});
也許有人會反駁, 那麼我可以寫在另外一個Thread內啊! 也沒錯, 但是相對的你就必須開多個Thread, 這樣一來就沒有確保執行緒安全了!
new Thread(new Runnable(){
public void run(){
//task 1
}
});
new Thread(new Runnable(){
public void run(){
//task 2
}
});
new Thread(new Runnable(){
public void run(){
//task 3
}
});
如上述例子 task1,task2,task3有存取到共同的資料結構, 則可能會產生concurrent的問題。 那也許會有人說(還真多人XD), 只要確保資料結構是同步的即可。
還是對的! 但是那個資料結構就必須使用Concurrent系列的, 或者自行實作synchronized,
效能相對會下降, 程式碼也會變得比較複雜。
不過使用方法必須對應到使用情境, 也不一定就是HandlerThread是萬用解藥,
畢竟它只有一個Thread, 要達到多核心必須使用多執行緒才能效能最大化,
之後可以參考如何使用ThreadPool跟如何使用AsyncTask