Android 中 Looper、Handler、Toast的交互理解

Looper維護一個消息循環,Handler爲這個消息隊列提供訪問的藉口。而Toast和Dialog中都有Handler成員變量,再初始化Handler時需要一個Looper,所以在子線程中需要調用Looper.prepare()準備消息隊列,否則會出錯!在子線程中也可以這樣調用Toast(把其加入到mainLooper中)

new Handler(Looper.getMainLooper()).post(new Runnable() {
@Override
public void run() {
// TODO Auto-generated method stub
Toast.makeText(MainActivity.this, error, 1000).show();
}
});


參考鏈接:http://blog.sina.com.cn/s/blog_78bb6d040101axl4.html


(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();

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);  
                }  
}  


發佈了31 篇原創文章 · 獲贊 2 · 訪問量 3萬+
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章