zoukankan      html  css  js  c++  java
  • Android中对于没有Looper的类,要使用Toast的问题

    因为Toast.java中初始化时需要创建一个Handler对象,而默认情况下,Handler会与其被定义时所在线程的Looper绑定,比如,在主线程中定义,其是与主线程的Looper绑定,那么,
    Toast.java中Handler handler = new Handler() 等价于new Handler(Looper.myLooper())。 这时,若Handler所在线程没有Looper,则会报错。

    1.一种解决方法

     1 public class TestService extends  Service {
     2       private Handler handler;
     3       @Override
     4       public IBinder onBind(Intent intent){
     5           return null;
     6       }
     7       
     8       @Override
     9       public void onCreate(){
    10           handler = new Handler(Looper.getMainLooper());                        
    11           handler.post(new Runnable() {  
    12                @Override  
    13                public void run() {  
    14                   Toast.makeText(getApplicationContext(), "Test",Toast.LENGTH_SHORT).show();  
    15                }  
    16           });
    17       }
    18   }

     也可以不需要创建Handler对象,直接三句搞定:

    1 Looper.prepare();
    2 Toast.makeText(getApplicationContext(), "Test",Toast.LENGTH_SHORT).show(); 
    3 Looper.loop();

     2.通过源码分析

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

    问题代码:

     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.getContext().getSystemService(Context.INPUT_METHOD_SERVICE);  
     8                     m.showSoftInput(editText, 0);  
     9                   
    10                     //Looper.prepare();  
    11                     Toast.makeText(Main.this, "show", Toast.LENGTH_LONG).show();  
    12                     //Looper.loop();  
    13                 }  
    14             }, 1000);  
    15         }  
    16 }  

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

    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.<init>(Handler.java:121)  
    E/AndroidRuntime( 1819):        at android.widget.Toast.<init>(Toast.java:68)  
    E/AndroidRuntime( 1819):        at android.widget.Toast.makeText(Toast.java:231)  

    Handler.java:121

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

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

    68    final Handler mHandler = new Handler(); 

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

     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行到底做了什么,如下所示:


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

     到此,距离真相的解开近了一大步,既然抛出了 RuntimeException,那么 mLooper 肯定是 null,但是为什么 Looper.myLooper() 会返回 null?继续进入到 Looper.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 是正确但非正常的结果。

  • 相关阅读:
    vue项目搭建
    轮播 删除中间的还是居中
    随内容增加,背景不设高度自适应,背景图不拉伸和变形
    緢点连接
    左侧背景,右侧数据,根据数据左侧背景自适应
    问题
    手机访问网站,点击手机号码直接拨打电话
    长度超出之后文字变成省略号
    css中vw,vh单位对于UC的兼容性问题
    线性渐变的兼容性写法
  • 原文地址:https://www.cnblogs.com/chenbin7/p/2640324.html
Copyright © 2011-2022 走看看