Android 多執行緒處理之多執行緒用法大集合 第 1 頁
handler.post(r)其實這樣並不會新起執行緒,只是執行的runnable裡的run()方法,卻沒有執行start()方法,所以runnable走的還是UI執行緒。
1.如果像這樣,是可以操作ui,但是run還是走在主執行緒,見打印出來的Log執行緒名字是main,說明是主執行緒。
這就是為什麼可以直接在run方法裡操作ui,因為它本質還是ui執行緒
handler.post(new Runnable(){
public void run(){
Log.e("當前執行緒:",Thread.currrentThread.getName());//這裡列印de結果會是main
setTitle("哈哈");
}
});
2.通過HandlerThread獲取到looper卻是可以新起執行緒,但是在這裡的run方法裡操作ui是不可能的,但是這顯然有個缺點,如果你執行多次post(r)方法其實走的還是HandlerThread執行緒。假如你執行5次,n次,其實還是一次並且它們是序列的。假如下載5張圖片,你會看到圖片是下完第一張,才會去下第二張。
實踐證明,只有是擁有主執行緒looper的handler才可以操作ui,而在主執行緒操作ui可以在handler的handlerMessage()方法中操作Ui,也可以在handler的post(r)的run方法裡操作Ui.
HandlerThread ht = new HandlerThread("handler thread");
ht.start();
handler = new Handler(ht.getLooper());
handler.post(new Runnable(){//這裡run()方法其實還是在等ht.start()呼叫
public void run(){
Log.e("當前執行緒:",Thread.currrentThread.getName());//這裡列印的會是handler thread
setTitle("哈哈");//這樣必定報錯
//android.view.ViewRoot$CalledFromWrongThreadException: Only the original thread that created a view hierarchy can touch its views.
}
});
這樣該怎麼辦呢,呵呵,可以無參構建一個handler。用這個handler來發送訊息和處理訊息,用上面的handler來開啟新執行緒。
mainHandler = new Handler(){
protecket void handlerMessage(Message msg){
setTitle("哈哈");//這樣就不會報錯啦
}
}
handler.post(new Runnable(){//這裡run()方法其實還是在等ht.start()呼叫
public void run(){
Log.e("當前執行緒:",Thread.currrentThread.getName());//這裡列印的會是handler thread
mainHandler.sendEmpetMessage();//用mainHandler來發送訊息
//setTitle("哈哈");//這樣必定報錯
//android.view.ViewRoot$CalledFromWrongThreadException: Only the original thread that created a view hierarchy can touch its views.
}
});
列印Log:
3.其實第2個方法顯得麻煩而且低效,用了2個handler,一個用來發起執行緒,一個用於處理訊息。發起執行緒的handler必須擁有looper,所以還要例項化一個HanderThread;而處理訊息的handler則不需要looper,因為它預設擁有主執行緒的looper,所以可以在這個handler處理ui。
其實可以只需要例項化一個handler,在主執行緒裡構建一個無參的handler,然後由它傳送和處理訊息。而建立執行緒的任務就不用handler了,直接用new Thread(r).start();然後在r的run()方法裡面處理邏輯事務。
用這樣的模式下載5張圖片,你就可能不會看到圖片一張挨著一張展示出來,可能第2張先出來,也可能同時出來3張,5條執行緒很隨機的。
private void loadImagesByThread(final String url,final int id){//通過Thread來new 出多個執行緒
new Thread(new Runnable(){
@Override
public void run() {
// TODO Auto-generated method stub
Log.e("當前執行緒:", ""+Thread.currentThread().getName());
Drawable drawable = null;
try {
drawable = Drawable.createFromStream(new URL(url).openStream(), "image.gif");
} catch (MalformedURLException e) {
// TODO Auto-generated catch block
e.printStackTrace();
} catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
Message msg = mainHandler.obtainMessage();
msg.what = 2012;
msg.arg1 = id;
msg.obj = drawable;
msg.sendToTarget();
}
}).start();
}
列印Log: