-
使用handler時遇到的問題查看全部
-
Handler查看全部
-
handler是android給我們提供用來更新UI的一套機制,也會死一套消息處理的機制,我們可以發送消息,也可以通過他處理消息查看全部
-
handler查看全部
-
handler:1、用于更新UI;2、是一套消息處理機制查看全部
-
/** * 與子線程相關的Handler */ public class MyThread extends Thread { public Handler handler; @Override public void run() { Looper.prepare();//消息隊列準備 handler = new Handler() { @Override public void handleMessage(Message msg) { System.out.println("currentThread:" + Thread.currentThread()); } }; Looper.loop();//循環處理消息 } }查看全部
-
程序運行-->ActivityThread.main()--[ Looper.prepareMainLooper() ] --[ prepare(boolean guitAllowed) ] -- [ threadLocal.set(new Looper(quitAllowed)) ] --[ new MessageQueue(quitAllowed) ]查看全部
-
1.不能直接在非UI線程直接更新UI(大多數時候)。<br> 2.每次創建Handler時需要給它綁定一個looper,如果是主線程不給定具體的looper則會綁定默認的looper。<br> 3.子線程運行時一定要調用start()方法。<br> 4.在某些特殊情況下在非UI線程是可以更新UI的//不推薦使用(當剛啟動Activity即onCreate里面此時onResume方法還沒有執行的時候可以,因為在線程中更新UI時會調用ViewParent.invalidateChild()方法檢查當前的Thread是否是UIThread,若為UIThread則可以更新(ViewParent是一個接口類,其實現是ViewRootImpl;invalidateChild()方法調用checkThread()方法來檢查線程是否為主線程)。ViewRootImp是在onResume方法中初始化的,所以只要在ViewRootImpl創建之前更新UI(在onCreate方法中創建線程并執行,此時還沒有初始化ViewRootImp),就可以逃避掉checkThread()的檢查進而更新UI。)查看全部
-
非UI線程能否更新UI --->剛啟動的時候,立即在非UI線程更新->不報錯。 --->休眠2s鐘以后,更新——————>報錯 更新UI-->會調用checkForRelayout()方法 -->invalidate()方法-->invalidate(true)方法,關注viewParent-->ViewRootImpl是ViewParent的實現類 --->p.invalidateChild()-->查看ViewRootImpl.invalidateChild()-->checkThread()方法-->判斷UI線程是否是當前線程,不想等拋出異常。 ViewRootImpl是onResume()方法才會創建。所以onCreate()方法中要延遲才可以。 handleResumeActivity()方法---》viewManager.addView()-->ViewRootImpl初始化。,關注viewParent-->ViewRootImpl是ViewParent的實現類查看全部
-
在非UI線程中是可以更新Ui的(在onCreate中創建新線程更新UI): 當剛啟動Activity即onCreate里面此時onResume方法還沒有執行的時候可以,因為在線程中更新UI時會調用ViewParent.invalidateChild()方法檢查當前的Thread是否是UIThread,若為UIThread則可以更新(ViewParent是一個接口類,其實現是ViewRootImpl;invalidateChild()方法調用checkThread()方法來檢查線程是否為主線程)。ViewRootImp是在onResume方法中初始化的,所以只要在ViewRootImpl創建之前更新UI(在onCreate方法中創建線程并執行,此時還沒有初始化ViewRootImp),就可以逃避掉checkThread()的檢查進而更新UI。查看全部
-
更新UI的4種方式(其實內部都是handler機制): 1.通過Handler的post方法(); 2.調用Handler.sendMessage()或Handler.sendEmptyMessage()方法,傳統的方法 3.重寫Activity中的runOnUIThread方法更新; 4.調用View自身的post(Runnable run)方法更新;查看全部
-
主線程和子線程之間如何相互通信: 1. 子線程向主線程發送消息,是通過調用主線程的Handler,發送信息給主線程的Looper,該Hnadler綁定了主線程的Looper。 2. 主線程向子線程發送消息,是通過調用子線程的Handler,發送信息給子線程的Looper,因此必須有子線程的Looper,為了防止Looper沒有初始化,所以通過HandlerThread類,來保證子線程的Looper再被主線程調用時已經初始化。查看全部
-
1.創建主線程的handler,并向子線程發送消息: //主線程的handler Handler handler = new Handler(){ public void handleMessage(android.os.Message msg) { Message message = new Message(); threadHandler.sendMessageDelayed(message, 1000);//向子線程發送消息 }; }; 2.創建子線程的handler,向主線程發送消息,要關聯一個threadHandler,threadHandler.getLooper()得到一個looper對象 HandlerThread handlerThread = new HandlerThread("handler thread"); handlerThread.start();//不要忘記調用start方法! //子線程的handler threadHandler = new Handler(handlerThread.getLooper()){ @Override public void handleMessage(Message msg) {//處理消息 Message message = new Message(); handler.sendMessageDelayed(message, 1000); } };查看全部
-
子線程與主線程之間的相互通信 1.子線程通過 HandlerThread的thread.getLooper()綁定, 2.在主線程的handler的handlerMessage中調用threadHandler.sendMessageDelay(msg,1000);向子線程發送消息。 3.在子線程中通過handler.sendMessageDelay(msg,1000);向主線程發送消息 4.在一個啟動點btn調用主線程的handler.sendEmptyMessage(int x); 5.在一個結束點btn調用handler.removeMessages(x);查看全部
-
1. Handler在創建的時候可以指定Looper,這樣通過Handler的sendMessage()方法發送出去的消息就會添加到指定Looper里面的MessageQueue里面去。在不指定Looper的情況下Handler綁定的是創建它的線程的Looper。如果這個線程的Looper不存在,程序將拋出"Can't create handler inside thread that has not called Looper.prepare()"。 2. HandlerThread繼承于Thread,所以它本質就是個Thread。與普通Thread的差別就在于,它自帶一個封裝好了的Looper成員變量。在其run()方法中,調用Looper.myLooper()獲得一個looper對象。 3. HandlerThread的用處 創建Handler時指定的looper,也可以是別的線程創建的。所以Handler中MessageQueue的輪詢不一定非要在創建Handler的線程進行,還可以在別的線程中進行。 這個時候我們就需要使用HandlerThread這個類的Looper,這樣消息的處理就在新創建的HandlerThread中進行。模擬異步處理,主線程給子線程發送消息,在子線程中處理比較耗時的操作。 mThread = new HandlerThread("Handler Thread");//為新創建的線程指定一個名字,HandlerThread線程獨有 mHandler = new Handler(mThread.getLooper()){ public void handleMessage(android.os.Message msg) {... }; };查看全部
舉報
0/150
提交
取消