Handler Looper Message 关系是什么?

分析 Handler

 

首先我们来分析分析一下 Handler 的用法,我们知道,要创建一个 Handler 对象非常的简单明了,直接进行 new 一个对象即可,但是你有没有想过,这里会隐藏着什么注意点呢。现在可以试着写一下下面的一小段代码,然后自己运行看看:

public class MainActivity extends ActionBarActivity {

    private Handler mHandler0;

    private Handler mHandler1;

    @Override

    protected void onCreate(Bundle savedInstanceState) {             
      super.onCreate(savedInstanceState); setContentView(R.layout.activity_main);


        mHandler0 = new Handler(); 
        new Thread(new Runnable() {
            @Override

            public void run() {

                mHandler1 = new Handler();

           }

          }).start();

}

这一小段程序代码主要创建了两个 Handler 对象,其中,一个在主线程中创建,而另外一个则在子线程中创建,现在运行一下程序,则你会发现,在子线程创建的Handler 对象竟然会导致程序直接崩溃,提示的错误竟然是Can't create handler inside thread that has not called Looper.prepare()于是我们按照 logcat 中所说,在子线程中加入 Looper.prepare(),即代码如下:

        new Thread(new Runnable() {
            @Override
            public void run() {
                Looper.prepare();
                mHandler1 = new Handler();
            }
        }).start();

再次运行一下程序,发现程序不会再崩溃了,可是,单单只加这句Looper.prepare()是否就能解决问题了。我们探讨问题,就要知其然,才能了解得更多。我们还是先分析一下源码吧,看看为什么在子线程中没有加Looper.prepare()就会出现崩溃,而主线程中为什么不用加这句代码?我们看下Handler()构造函数:

public Handler() {

this(null, false);
}

构造函数直接调用 this(null, false),于是接着看其调用的函数,

public Handler(Callback callback, boolean async) { 
    if (FIND_POTENTIAL_LEAKS) {
        final Class<? extends Handler> klass = getClass();

        if ((klass.isAnonymousClass() || klass.isMemberClass() || kla ss.isLocalClass())             
        &&(klass.getModifiers() & Modifier.STATIC) == 0) {

        Log.w(TAG, "The following Handler class should be static o r leaks might occur: "            
         +klass.getCanonicalName());

        }

   }




    mLooper = Looper.myLooper(); 
    if (mLooper == null) {
        throw new RuntimeException(

        "Can't create handler inside thread that has not called Lo oper.prepare()");

    }
    mQueue = mLooper.mQueue;
    mCallback = callback;
    mAsynchronous = async;
}

不难看出,源码中调用了 mLooper = Looper.myLooper()方法获取一个 Looper对象,若此时 Looper 对象为 null,则会直接抛出一个“Can't create handlerinside thread that has not called Looper.prepare()”异常

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值