博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
Android的Handler,Message,Looper的原理详解
阅读量:6257 次
发布时间:2019-06-22

本文共 5013 字,大约阅读时间需要 16 分钟。

hot3.png

先说说Looper类:Looper就是为每一个线程创建一个Looper对象,维护一个MessageQueue,并循环从MessageQueue中取出消息并分发给Handler执行。下面是Looper源码中如何使用的一个示例

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.prepare干了些什么事

public static void prepare() {

        prepare(true);

    }

    //判断当前线程是否有Looper对象,没有则new一个并放到ThreadLocal中

    private static void prepare(boolean quitAllowed) {

        if (sThreadLocal.get() != null) {

            throw new RuntimeException("Only one Looper may be created per thread");

        }

        sThreadLocal.set(new Looper(quitAllowed));

    }

至于这个地方为什么要用ThreadLocal来存储Looper对象呢:

我们知道ThreadLocal是与线程相关的一个类,系统规定每个线程只能有一个Looper对象,也就是说只能由一个MessageQueue对象,这样便于维护消息队列。如果我已经prepare一次了,那么我的当前线程中是有Looper对象的,并存储在当前线程变量内。当你再在此线程prepare时,ThreadLocal对象会获取当前线程是否有Looper对象,如果有,直接抛异常,这是系统所不允许的,

static final ThreadLocal<Looper> sThreadLocal = new ThreadLocal<Looper>();

因此ThreadLocal用来存储Looper对象,设计的真鸡贼,不知道大家还有没有别的方式来控制一个线程只有一个Looper对象。看到很多博客都说ThreadLocal是用来并发控制资源共享的,ThreadLocal存储每个线程对资源的副本,再次本人觉得纯属扯淡,你去看看ThreadLocal的例子,如果ThreadLocal存储副本,你丫的把“主本”给我找出来?通过Looper就可以看出来ThreadLocal只是存储当前线程的相关变量。

创建完Looper对象,并初始化了MessageQueue对象,下面该进入不断循环,取出消息了:

public static void loop() {

        final Looper me = myLooper();//你看,每次都是myLooper获取当前线程的Looper对象,免得获取到别的线程的了

        if (me == null) {

            throw new RuntimeException("No Looper; Looper.prepare() wasn't called on this thread.");

        }

        final MessageQueue queue = me.mQueue;

        // Make sure the identity of this thread is that of the local process,

        // and keep track of what that identity token actually is.

        Binder.clearCallingIdentity();

        final long ident = Binder.clearCallingIdentity();

        for (;;) {//死循环吧

            Message msg = queue.next(); // 从队列中获取下一个

            if (msg == null) {

                // No message indicates that the message queue is quitting.

                return;

            }

            // This must be in a local variable, in case a UI event sets the logger

            Printer logging = me.mLogging;

            if (logging != null) {

                logging.println(">>>>> Dispatching to " + msg.target + " " +

                        msg.callback + ": " + msg.what);

            }

            //获取到消息之后分发给HAndler处理,其实这个Handler就是target

            msg.target.dispatchMessage(msg);

            if (logging != null) {

                logging.println("<<<<< Finished to " + msg.target + " " + msg.callback);

            }

            // Make sure that during the course of dispatching the

            // identity of the thread wasn't corrupted.

            final long newIdent = Binder.clearCallingIdentity();

            if (ident != newIdent) {

                Log.wtf(TAG, "Thread identity changed from 0x"

                        + Long.toHexString(ident) + " to 0x"

                        + Long.toHexString(newIdent) + " while dispatching to "

                        + msg.target.getClass().getName() + " "

                        + msg.callback + " what=" + msg.what);

            }

            msg.recycle();//回收Message对象,重复利用就在此,哈哈。。下面会讲

        }

    }

至此,Looper的主要工作讲完了。

接着是Message,就是携带了一些数据,被handler带着飞。重复利用问题以及链表的结构。

先看看都带了什么数据:

/*package*/ int flags;

    /*package*/ long when;//时间而已

    

    /*package*/ Bundle data;//数据

    

    /*package*/ Handler target;     //Handler,表明我这个Message要被哪个Handler处理,当然是谁发送我,谁处理,发送我的时候给我赋值

    

    /*package*/ Runnable callback;  //post(Runnable r)函数使用 

    

    // sometimes we store linked lists of these things

    /*package*/ Message next;//链接下一个Message

再看看重复利用:

public void recycle() {

        clearForRecycle();//把要回收的这个Message数据清空

        synchronized (sPoolSync) {

            if (sPoolSize < MAX_POOL_SIZE) {//不超过50个,源码的常量

                next = sPool;

                sPool = this;//关键在这,又把这个对象赋给了静态的sPool引用,gc不会回收此Message

                sPoolSize++;

            }

        }

    }

 public static Message obtain() {

        synchronized (sPoolSync) {

            if (sPool != null) {

                Message m = sPool;

                sPool = m.next;

                m.next = null;

                sPoolSize--;

                return m;

            }

        }

        return new Message();

    }

看吧,我们obtain的时候会利用刚才回收掉的Message,否则才new Message();

再看MessageQueue,他里面并不真正的使用数组的形式去存储一系列的Message,而是通过Message的next使用链表的形式维护Message队列,只要能拿到链表的第一个Message就可以展开一系列的循环读取Message。通过next()方法获取。代码就不贴了。

最后是Handler类介绍:创建一个Handler对象,一般都会重写handleMesage方法。去处理我们发送过来的消息,handleMessage何时被调用呢,看下

 public void dispatchMessage(Message msg) {

        if (msg.callback != null) {

            handleCallback(msg);//处理handler.post(Runnable r)的

        } else {

            if (mCallback != null) {

                if (mCallback.handleMessage(msg)) {

                    return;

                }

            }

            handleMessage(msg);//调用了吧

        }

    }

那么dispatchMessage的调用呢,看前面的Looper.loop(),娶到一个Message之后,通过Message 的target(Handler)调用到的,这样处理的过程。

再看看Handler的发送:

public boolean sendMessageAtTime(Message msg, long uptimeMillis) {

        MessageQueue queue = mQueue;

        if (queue == null) {

            RuntimeException e = new RuntimeException(

                    this + " sendMessageAtTime() called with no mQueue");

            Log.w("Looper", e.getMessage(), e);

            return false;

        }

        return enqueueMessage(queue, msg, uptimeMillis);

    }

    private boolean enqueueMessage(MessageQueue queue, Message msg, long uptimeMillis) {

        msg.target = this;//这个很重要就是把此Handler对象直接赋给此Message的target,

        if (mAsynchronous) {

            msg.setAsynchronous(true);

        }

        return queue.enqueueMessage(msg, uptimeMillis);//这个就是把此Message对象加入MessageQueue

    }

这样,整个流程就走完了,发送到加入队列,再取,在处理的流程。

发送是Handler,加入队列是MessageQueue,取是Looper(内部也是MesageQueue取的,不过是有Looper维护此队列而已),处理是Handler。

Activity主线程,在程序中的main方法已经prepare了,并且loop了,所以我们没必要创建Looper对象。

在我们自己创建的子线程中,我们要想处理消息,必须Looper.prepare,并且loop()。

转载于:https://my.oschina.net/u/1254864/blog/213705

你可能感兴趣的文章
Linux学习笔记——程序包管理之yum
查看>>
SqlServer转换为Mysql的一款工具推荐(mss2sql)
查看>>
go装饰模式,一个屌丝撸管的故事
查看>>
学习设计模式——命令模式
查看>>
【POJ】第一章 C/C++语言概述
查看>>
如何封装自己的js类库
查看>>
项目管理小小知识点总结
查看>>
ASP.NET之Javascript脚本的应用
查看>>
vlan间的互通
查看>>
ldconfig详解
查看>>
VBScript 页面的简单样例
查看>>
用c语言指针实现给整形数组冒泡排序
查看>>
ORA-01075,ORA-09925 Read-only file system问题一例
查看>>
Script:收集介质恢复诊断信息
查看>>
SocketIO 随笔
查看>>
Maven学习三之新建maven项目
查看>>
HTML5本地存储-localStorage如何实现定时存储
查看>>
LAMP之Centos6.5安装配置Apache(二)
查看>>
Tomcat集群
查看>>
shell脚本中输出带颜色字体实例分享及chrony时间同步
查看>>