android 开发Handler源码剖析

Android的消息机制主要是Handler的运行机制,而讲Handler的机制,又需要和MessageQueue和Looper结合。MessageQueue中文意思是消息队列,虽说叫队列,但是其内部结构并不是队列组成的,而是采用了单链表的形式来存储消息。MessageQueue只是负责存储消息,并不处理消息(这里指消息的轮训),Looper刚好弥补了这个空缺。我在知道,Handler创建的时候,会默认为我们创建一个Looper对象,那么如何获取当前的Looper呢,这里就使用到了一个TheadLocal的概念,TheadLocal可以轻松的获取当前使用的Looper。

Handler的使用:

Handler 主要有两种用法:( 1 )调度消息和 runnable 对象,并在某一时间点执行。( 2 )依次存放分属不同线程的行为。

线程在默认的时候是没有Looper与之相关联。在线程中,我们可以通过调用prepare 方法来启动一个消息loop,调用loop方法来通知Looper来处理消息,直至结束。Looper提供了很多的静态方法来与线程、消息队列进行交互。一个线程最多允许创建一个Looper。

以下是SDK文档中介绍的在线程中使用handler的一种方法:

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

注意这里,官方的例子是和Looper关联了,那么如果我们不关联Looper的话会怎么样呢

 mRunnable = new  Runnable(){
           @Override
           public void run() {
              // TODO Auto-generated method stub    
              String s = mHandler.getLooper().getThread().getName();
              Log.e("handler", s);
              mHandler.postDelayed(mRunnable, 1000);
           }};
Thread t  = new Thread(mRunnable,"thread");
        t.start();

运行结果:

ERROR/handler(1630): main,可以发现我们创建了一个名为“thread”的线程,病通过handler发送消息,我们创建的handler自动和UI线程的Looper关联上了。

那么,可不可以在主线程中定义handler,在代码中动态改变handler使之与其他线程的looper相关联呢?是可以的。

mRunnable = new  Runnable(){
              @Override
              public void run() {
                  // TODO Auto-generated method stub
                  Looper.prepare();
                  mHandler = new Handler(Looper.myLooper());
                  String s = mHandler.getLooper().getThread().getName();
                  Log.e("handler", s);
                  mHandler = new Handler(Looper.getMainLooper());
                   s = mHandler.getLooper().getThread().getName();
                  Log.e("handler", s);        
              }};
    Thread t  = new Thread(mRunnable,"subthread");
            t.start();

运行结果:

ERROR/handler(4049):subthread

ERROR/handler(4049): main

不过我们Android为我们提供了一个包含Looper的Thead类,HandlerThread HandlerThread自带了一个Lopper,

 handlerThread=new HandlerThread("handlerThead");
        handlerThread.start();
        handler=new Handler(handlerThread.getLooper());
        handler.post(new Runnable() {
            @Override
            public void run() {
                String s=handler.getLooper().getThread().getName();
                Log.d("test",s);
            }
        });

注:android认为在非线程中操作UI界面是不安全的,因此禁止在其它线程中修改UI界面。解决的方法有两种,一是通过在主线程中定义的handler更新界面,二是直接调用被修改的view的postInvalidate方法刷新单个view。

其实上面的描述是不准确的,实际上在activity第一次执行的时候,是可以在子线程直接更新主线程数据的,条件是不能执行Thread.sleep(),

原因是什么呢?

1.当刚启动Activity即onCreate里面此时onResume方法还没有执行的时候可以,因为在线程中更新UI时会调用ViewParent.invalidateChild()方法检查当前的Thread是否是UIThread,若为UIThread则可以更新(ViewParent是一个接口类,其实现是ViewRootImpl;invalidateChild()方法调用checkThread()方法来检查线程是否为主线程)。ViewRootImp是在onResume方法中初始化的,所以只要在ViewRootImpl创建之前更新UI(在onCreate方法中创建线程并执行,此时还没有初始化ViewRootImp),就可以逃避掉checkThread()的检查进而更新UI。 2.-->刚启动的时候,立即在非UI线程更新->不报错(onResume还没有执行) --->休眠2s钟以后,更新——————>报错

最后贴上一张图

分析完了Android消息机制的流程,那么我们接下来分别理解一些重要的概念。主要从以下几个方面增强理解,Handler,MessageQueue,Looper和Threadloacal.

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏C#

HTTP在.NET中的一些应用和解析

     谈到HTTP协议(超文本传输协议),HTTP协议是一个基于请求与响应模式的、无状态的、应用层的协议,常基于TCP的连接方式,HTTP1.1版本中给...

2019
来自专栏子勰随笔

Android简单实现的多线程下载模块

2116
来自专栏Golang语言社区

golang(GO语言)http详解简单基础

因为好像长时间的写PHP可能大家感觉烦躁了,所以写一点golang的东西大家可以拿去玩玩,golang在web开发中让你得心应手,其实也是很好的东西,只要你玩进...

5146
来自专栏为数不多的Android技巧

你真的了解AsyncTask?

虽说现在做网络请求有了Volley全家桶和OkHttp这样好用的库,但是在处理其他后台任务以及与UI交互上,还是需要用到AsyncTask。但是你真的了解Asy...

1022
来自专栏java一日一条

每个 Android 开发者必须知道的消息机制问题总结

不能,一个线程对应一个Looper对象,通过ThreadLocal保证一个线程只有一个Looper与之对应,如果多次调用Looper.prepare();则会抛...

1013
来自专栏Android中高级开发

Android开发之漫漫长途 Ⅶ——Android消息机制(Looper Handler MessageQueue Message)

该文章是一个系列文章,是本人在Android开发的漫漫长途上的一点感想和记录,我会尽量按照先易后难的顺序进行编写该系列。该系列引用了《Android开发艺术探索...

982
来自专栏我的博客

Laravel 权限控制基础之Gate 和Policy

policy和Gate php artisan make:policy PostPolicy –model=Post //特定model //AuthServi...

3246
来自专栏耕耘实录

批量创建用户并使用sudo和ACL来控制用户权限

版权声明:本文为耕耘实录原创文章,各大自媒体平台同步更新。欢迎转载,转载请注明出处,谢谢。

1014
来自专栏Phoenix的Android之旅

AsyncTask的限制你了解吗?

· executeOnExecutor(Executor exec, Params …)

942
来自专栏LIN_ZONE

windows环境下 php 将office文件(word/excel/ppt)转化为pdf(转)

2.使用office提供的服务 (注:这在windows服务器上,并且服务器上面安装了版本比较高的office)

1452

扫码关注云+社区

领取腾讯云代金券