下面是一段大家都比較熟悉的代碼:
複製代碼 代碼如下:
Handler handler = new Handler();
handler.post(myThread);
//使用匿名內部類建立一個線程myThread
Runnable mythread = new Runnable() {
public void run() {
}
};
一開始,相信很多人都以為myThread中的run()方法會在一個新的線程中運行,但事實並非如此。
上述代碼中的handler並沒有調用線程myThread的start()方法,而是直接調用了run()方法,這也就意味著實際上並沒有建立一個新的線程,只是在當前線程中調用run()方法而已。
這牽扯出一個問題,如果我們將一個很耗時的操作放到了run()方法內,然後使用一個Handler對象將該線程post到線程隊列。原本我們希望將這些耗時操作放到另外一個線程中,以免影響當前進程。但實際上卻恰恰相反:post()以下的那些代碼必須等到run()方法執行完畢後才能繼續執行。如當前線程為主線程,那麼主程式便會處於硬直狀態。
那麼應該如何去實現真正的多線程呢?
一種最簡便的方法就是直接利用JAVA中的實現多線程的方法,即建立一個Thread對象,然後調用start()方法。
還有另外一種方法,代碼如下:
複製代碼 代碼如下:
//HandlerThread建立了一個新線程,它包含一個Looper
HandlerThread handlerThread = new HandlerThread("handler_Thread");
handlerThread.start();//啟動一個線程
MyHandler myHandler = new MyHandler(handlerThread.getLooper());//使用新線程的Looper建立一個Handler
//此時MyHandler便與一個新線程綁定到一起了
Message msg = myHandler.obtainMessage();
msg.sendToTarget();//將message壓入提供message的Handler的訊息佇列中
//建立一個Handler的子類
class MyHandler extends Handler {
MyHandler(Looper looper) {
super(looper);
}
public void handleMessaage(Message msg) {
//處理訊息的代碼
}
}
今天我又仔細的看了一下Android文檔,發現android雖然支援上兩種建立線程的方法,但是它有一個規則:
原文是:it violates the second rule of the single-threaded model: do not access the Android UI toolKit from outside the UI thread.
大概意思就是:Android不支援在UI線程以外的線程中修改UI控制項。比如給一個Textview設定文字,這樣的操作便不能放在UI線程以外的線程內執行,否則便會出現異常。