Can't create handler inside thread that has not called Looper.prepare()


問題: Can't create handler inside thread that has not called Looper.prepare()

1,在報錯的方法前加Looper.prepare(); 
     方法末尾加Looper.loop();

 

2,問題原因:
     在android的多線程開發中,比如asyncTask,在其doInBackground()方法,調用了更新UI的方法。

     解決辦法:
     把更新UI的操作,放到消息處理器中處理;在doInBackground()方法中發送更新消息:

     Handler updateDate = new Handler(){
         @Override
         public void handleMessage(Message msg) {
             switch(msg.what){
                 case LOADING_FINISHED:
                     listView.setAdapter(gameAdapter);
                     break;
              }
          }
      };

3,
Toast和Looper。Handler消息循環機制。
 

(1) Looper類別用來為一個線程開啟一個消息循環。默認情況下Android中新誕生的線程是沒有開啟消息循環的。(主線程除外,主線程系統會自動為其創建Looper對象,開啟消息循環)

Looper對象通過MessageQueue來存放消息和事件。一個線程只能有一個Looper,對應一個MessageQueue。

 

(2) 通常是通過Handler對象來與Looper交互的。Handler可看做是Looper的一個接口,用來向指定的Looper發送消息及定義處理方法。

默認情況下Handler會與其被定義時所在線程的Looper綁定,比如,在主線程中定義,其是與主線程的Looper綁定。

mainHandler = new Handler() 等價於new Handler(Looper.myLooper()).

Looper.myLooper():Return the Looper object associated with the current thread 獲取當前進程的looper對象。

還有一個類似的 Looper.getMainLooper() 用於獲取主線程的Looper對象。

 

(3) 在非主線程中直接new Handler() 會報如下的錯誤:

E/AndroidRuntime( 6173): Uncaught handler: thread Thread-8 exiting due to uncaught exception
E/AndroidRuntime( 6173): java.lang.RuntimeException: Can't create handler inside thread that has not called Looper.prepare()

原因是非主線程中默認沒有創建Looper對象,需要先調用Looper.prepare()啟用Looper。

 

(4) Looper.loop(); 讓Looper開始工作,從消息隊列里取消息,處理消息。

注意:寫在Looper.loop()之后的代碼不會被執行,這個函數內部應該是一個循環,當調用mHandler.getLooper().quit()后,loop才會中止,其后的代碼才能得以運行。

 

(5) 基於以上知識,可實現主線程給子線程(非主線程)發送消息。

 

Toast或者Dialog中都有一個Handler的成員變量,在初始化時都會跟着初始化,而Toast或者Dialog中的Handler都需要一個Looper,所以需要在包含該Toast或者Dialog的線程中(如下面的Timer線程)初始化Looper。Looper.prepare();

問題代碼:

Java代碼  
private Handler myHandler = new Handler() {  
        public void handleMessage(Message msg) {  
                                Timer timer = new Timer();  
                timer.schedule(new TimerTask() {  
                    @Override  
                    public void run() {  
                        InputMethodManager m = (InputMethodManager) editText  
                                .getContext().getSystemService(  
                                        Context.INPUT_METHOD_SERVICE);  
                        m.showSoftInput(editText, 0);  
                        //  
                        Looper.prepare();  
                        Toast.makeText(Main.this, "show", Toast.LENGTH_LONG).show();  
                        Looper.loop();  
                    }  
                }, 1000);  
                }  
}  
 

 

Toast 和 Looper,一個屬於 android.widget,一個屬於 android.os,兩個貌似聯系不怎么緊密的類,卻通過下面這個異常聯系到了一起:

Java代碼  
E/AndroidRuntime( 1819): java.lang.RuntimeException: Can't create handler inside thread that has not called Looper.prepare()  
E/AndroidRuntime( 1819):        at android.os.Handler.(Handler.java:121)  
E/AndroidRuntime( 1819):        at android.widget.Toast.(Toast.java:68)  
E/AndroidRuntime( 1819):        at android.widget.Toast.makeText(Toast.java:231)  
 

Handler.java:121

Java代碼  
         mLooper = Looper.myLooper();  
         if (mLooper == null) {  
             throw new RuntimeException(  
             "Can't create handler inside thread that has not called Looper.prepare()");}  
 

Toast.java:68 ——>成員變量,在初始化時會跟着初始化

Java代碼  
     final Handler mHandler = new Handler();  
由以上的錯誤信息可以看出:程序要創建 handler,但是發現Looper.prepare還沒有被調用。通過 Android SDK 中的Reference可以看到,Looper、Handler 的調用是非常有講究的,如下面示例代碼:

 

Java代碼  
class LooperThread extends Thread {  
    public Handler mHandler;  
   
    public void run() {  
        Looper.prepare();  
        mHandler = new Handler() {  
            public void handleMessage(Message msg) {  
                // process incoming messages here  
            }  
        };  
        Looper.loop();  
    }  
}  
 言歸正題,繼續尋找 Toast、Looper 和 Handler 三者之間的聯系,也許謎底就能解開了。欲揭謎底,從源碼入手是一條捷徑。

Toast.java 的第231行的代碼是創建一個新的Toast實例,而實例化的過程中,就需要執行第68行,也就是聲明並創建Handler(成員變量)的實例。那么來看Handler.java的第121行到底做了什么,如下所示:

Java代碼  
         mLooper = Looper.myLooper();  
         if (mLooper == null) {  
             throw new RuntimeException(  
             "Can't create handler inside thread that has not called Looper.prepare()");}  
 到此,距離真相的解開近了一大步,既然拋出了 RuntimeException,那么 mLooper 肯定是 null,但是為什么 Looper.myLooper() 會返回 null?繼續進入到 Looper.java 中尋根究底。

 

Java代碼  
  
public static final Looper myLooper() {  
    return (Looper)sThreadLocal.get();  
}  
 以上就是 myLooper() 方法的真實面貌,通過注釋可以看出問題的真正原因在於當前線程並沒有綁定 Looper,返回為 null 是正確但非正常的結果。


 原文:https://blog.csdn.net/SamLee1989/article/details/8289631  謝謝!


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM