`

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代码  收藏代码
  1. private Handler myHandler = new Handler() {  
  2.         public void handleMessage(Message msg) {  
  3.                                 Timer timer = new Timer();  
  4.                 timer.schedule(new TimerTask() {  
  5.                     @Override  
  6.                     public void run() {  
  7.                         InputMethodManager m = (InputMethodManager) editText  
  8.                                 .getContext().getSystemService(  
  9.                                         Context.INPUT_METHOD_SERVICE);  
  10.                         m.showSoftInput(editText, 0);  
  11.                         //  
  12.                         Looper.prepare();  
  13.                         Toast.makeText(Main.this"show", Toast.LENGTH_LONG).show();  
  14.                         Looper.loop();  
  15.                     }  
  16.                 }, 1000);  
  17.                 }  
  18. }  

 

 

Toast 和 Looper,一个属于 android.widget,一个属于 android.os,两个貌似联系不怎么紧密的类,却通过下面这个异常联系到了一起

Java代码  收藏代码
  1. E/AndroidRuntime( 1819): java.lang.RuntimeException: Can't create handler inside thread that has not called Looper.prepare()  
  2. E/AndroidRuntime( 1819):        at android.os.Handler.<init>(Handler.java:121)  
  3. E/AndroidRuntime( 1819):        at android.widget.Toast.<init>(Toast.java:68)  
  4. E/AndroidRuntime( 1819):        at android.widget.Toast.makeText(Toast.java:231)  

 

Handler.java:121

Java代码  收藏代码
  1. 119        mLooper = Looper.myLooper();  
  2. 120        if (mLooper == null) {  
  3. 121            throw new RuntimeException(  
  4. 122            "Can't create handler inside thread that has not called Looper.prepare()");}  

 

Toast.java:68 ——>成员变量,在初始化时会跟着初始化

Java代码  收藏代码
  1. 68    final Handler mHandler = new Handler();  

由以上的错误信息可以看出:程序要创建 handler,但是发现Looper.prepare还没有被调用。通过 Android SDK 中的Reference可以看到,Looper、Handler 的调用是非常有讲究的,如下面示例代码

 

Java代码  收藏代码
  1. class LooperThread extends Thread {  
  2.     public Handler mHandler;  
  3.    
  4.     public void run() {  
  5.         Looper.prepare();  
  6.         mHandler = new Handler() {  
  7.             public void handleMessage(Message msg) {  
  8.                 // process incoming messages here  
  9.             }  
  10.         };  
  11.         Looper.loop();  
  12.     }  
  13. }  

 言归正题,继续寻找 Toast、Looper 和 Handler 三者之间的联系,也许谜底就能解开了。欲揭谜底,从源码入手是一条捷径。

Toast.java 的第231行的代码是创建一个新的Toast实例,而实例化的过程中,就需要执行第68行,也就是声明并创建Handler(成员变量)的实例。那么来看Handler.java的第121行到底做了什么,如下所示:

Java代码  收藏代码
  1. 119        mLooper = Looper.myLooper();  
  2. 120        if (mLooper == null) {  
  3. 121            throw new RuntimeException(  
  4. 122            "Can't create handler inside thread that has not called Looper.prepare()");}  

 到此,距离真相的解开近了一大步,既然抛出了 RuntimeException,那么 mLooper 肯定是 null,但是为什么 Looper.myLooper() 会返回 null?继续进入到 Looper.java 中寻根究底。

 

Java代码  收藏代码
  1. /** 
  2.  * Return the Looper object associated with the current thread.  Returns 
  3.  * null if the calling thread is not associated with a Looper. 
  4.  */  
  5. public static final Looper myLooper() {  
  6.     return (Looper)sThreadLocal.get();  
  7. }  

 以上就是 myLooper() 方法的真实面貌,通过注释可以看出问题的真正原因在于当前线程并没有绑定 Looper,返回为 null 是正确但非正常的结果。

 

http://jeff-pluto-1874.iteye.com/blog/869710

分享到:
评论

相关推荐

    android service thread toast handler

    `Handler`是Android消息处理机制的一部分,它与`Looper`和`Message`协同工作,用于在不同的线程之间传递消息。在主线程中创建`Handler`实例,然后在子线程中通过该`Handler`发送`Message`,这样就可以在子线程完成...

    Handler与AsyncTask,Looper使用示例

    默认情况下,主线程有一个运行的`Looper`,而子线程需要手动调用`Looper.prepare()`和`Looper.loop()`来启动消息循环。 使用示例: ```java // 创建Handler实例 Handler handler = new Handler() { @Override ...

    了解Android核心:Looper,Handler和HandlerThread

    在Android开发中,理解核心组件如Looper、Handler和HandlerThread对于构建高效且响应式的用户界面至关重要。这些组件共同协作,确保UI线程不被阻塞,从而提供流畅的用户体验。 首先,主线程(也称为UI线程)是...

    Android Handler 线程 示例代码

    在Android开发中,`Handler`、`Thread`、`Toast`、`消息队列`以及`线程`是实现UI交互和异步处理的关键概念。本文将深入探讨这些概念,并结合给定的"Android Handler 线程 示例代码"进行详细讲解。 首先,我们来看`...

    android Handler的使用(一)

    `Handler`的基本原理是基于消息队列(Message Queue)和Looper的循环处理。在Android系统中,每个线程都有一个Looper对象,但默认只有主线程(UI线程)的Looper已经启动并准备处理消息。当创建一个`Handler`实例时,...

    自己整理的Toast(吐司)工具类.txt

    - `Handler` 和 `Looper`: 实现消息循环机制,用于在主线程更新UI。 - `Gravity`: 控制Toast的位置。 - `Toast`: Android系统提供的用于显示短消息的组件。 2. **静态成员变量** - `handler`: `Handler`实例,...

    android service toast 01

    2. 使用Handler:创建一个与主线程关联的Handler,然后通过Handler发送消息来显示Toast。这种方式适用于Service没有与特定Activity绑定的情况。 ```java // 在Service的onCreate()中 Handler mainHandler = new ...

    Android Handler的使用1

    主线程默认有Looper,其他线程需要手动调用`Looper.prepare()`和`Looper.loop()`来启动消息循环。 4. **消息队列的生命周期**:Handler依赖于Looper,而Looper又依赖于其所在的线程。因此,当线程结束时,Handler...

    如何在线程中正确使用Toast显示提示信息

    Handler handler = new Handler(Looper.getMainLooper()); handler.post(new Runnable() { public void run() { Toast.makeText(MainAct.messtv.getContext(),mes,Toast.LENGTH_LONG).show(); } }); ``` 在线程...

    android Toast设置显示时间效果

    Handler handler = new Handler(Looper.getMainLooper()); handler.postDelayed(new Runnable() { @Override public void run() { super.run(); // 使用自定义的时长显示Toast if (!View.GONE.equals(getView...

    Android 五种不同的Toast效果-IT计算机-毕业设计.zip

    在Android应用开发中,`Toast`是一个非常常用的通知机制,它能短暂地显示一条消息,然后自动消失。这个毕业设计的源码示例探讨了五种不同的`Toast`效果,帮助学习者深入理解如何定制和扩展`Toast`的显示方式。下面...

    Android studio 提示框Toast 弹出框AlertDialog 多种提示方法

    Handler handler = new Handler(Looper.getMainLooper()); handler.post(new Runnable() { @Override public void run() { //放在UI线程弹Toast Toast.makeText(Setting.this, 正在更新请稍后, Toast.LENGTH_...

    HandlerThreadDemo

    `Looper` 是一个消息循环,负责接收并分发`Handler`发送的消息。一旦`HandlerThread`启动,它的`Looper`就开始持续运行,等待消息。 在`HandlerThreadDemo`中,我们通常会做以下几步: 1. **创建 HandlerThread:*...

    Android Service中使用Toast无法正常显示问题的解决方法

    为了解决这个问题,我们可以利用Handler和Looper来在主线程上执行显示Toast的操作。 以下是使用Handler解决Service中Toast无法显示问题的步骤: 1. 在Service的适当位置(如onCreate())创建一个新的Handler实例,...

    Android中使用Toast.cancel()方法优化toast内容显示的解决方法

    `Handler` 负责在主线程执行更新和显示`Toast`的操作,以确保UI操作的正确性。`synchronized (synObj)` 用于同步对`toast`对象的访问,防止多线程并发问题。当调用`showMessage`方法时,首先检查是否存在已创建的`...

    安卓Andriod源码——五种效果的Toast.zip

    因此,需要确保Toast的创建和显示都在主线程中进行,可以使用`runOnUiThread`或`Looper.getMainLooper().post`来保证这一点。 了解并实现这些不同的Toast效果,开发者不仅可以提高应用的用户体验,还能更好地掌握...

    android异步消息机制 从源码层面解析(2)

    这种机制的核心组件包括Handler、Looper和Message。在Android中,主线程负责与用户交互,而其他后台任务则在单独的线程中执行。异步消息机制允许后台线程与主线程通信,更新UI。 首先,我们来深入理解AsyncTask,它...

    Android Toast通知用法实例详解

    在Android开发中,`Toast`是一种轻量级的通知机制,用于在短时间内向用户显示简短的信息,而不会中断用户的操作。这些信息会在指定的时间后自动消失。以下是对Android `Toast`通知用法的详细说明: 1. **默认用法**...

    点击两次退出程序

    在本案例中,实现这一功能主要依赖于Android的消息处理机制——Handler,以及Boolean变量的管理和计时器。 首先,我们需要理解Handler的基本原理。Handler是Android中的一个关键组件,它与Looper和Message协同工作...

Global site tag (gtag.js) - Google Analytics