首页 > 学院 > 开发设计 > 正文

Handler发送消息后消息队列的处理

2019-11-09 14:35:27
字体:
来源:转载
供稿:网友

在Android中使用Handler不仅可以直接post发送消息,还可以通过postDelayed设定延迟时间,来延迟执行消息任务。那么这后面的机制是如何处理的呢?

首先查看Handler的源码,会发现postpostDelayed内部均调用的同一方法sendMessageDelayed

public final boolean post(Runnable r){ return sendMessageDelayed(getPostMessage(r), 0);}public final boolean postDelayed(Runnable r, long delayMillis){ return sendMessageDelayed(getPostMessage(r), delayMillis);}

在这里的不同是当调用post时,延迟时间传入的是0。

下一步,sendMessageDelayed中:

public final boolean sendMessageDelayed(Message msg, long delayMillis){ if (delayMillis < 0) { delayMillis = 0; } return sendMessageAtTime(msg, SystemClock.uptimeMillis() + delayMillis);}

将小于0的异常延迟时间置为0,接下来将延迟时间与SystemClock.uptimeMillis()相加,最终调用sendMessageAtTime。 这一步的时间转换,将延迟时间这个相对值,转化为了系统启动后的一个绝对值时间,之后framework中的处理均是在绝对值上进行比较。

再继续追踪源码,handler最终会调用MessageQueue中的方法,将消息添加到队列中:

PRivate boolean enqueueMessage(MessageQueue queue, Message msg, long uptimeMillis) { msg.target = this; if (mAsynchronous) { msg.setAsynchronous(true); } return queue.enqueueMessage(msg, uptimeMillis); // 添加到消息队列}

进一步分析MessageQueue中的enqueueMessage方法,代码比较长,但是逻辑不复杂,添加到队列的代码主要是这一部分:

boolean enqueueMessage(Message msg, long when) { // ... synchronized (this) { // ... msg.markInUse(); msg.when = when; Message p = mMessages; boolean needWake; if (p == null || when == 0 || when < p.when) { // 将消息添加到队列开始 // New head, wake up the event queue if blocked. msg.next = p; mMessages = msg; needWake = mBlocked; } else { // 将消息添加到队列中间某处 // Inserted within the middle of the queue. Usually we don't have to wake // up the event queue unless there is a barrier at the head of the queue // and the message is the earliest asynchronous message in the queue. needWake = mBlocked && p.target == null && msg.isAsynchronous(); // 一个典型的链表插入节点方法 Message prev; for (;;) { prev = p; p = p.next; if (p == null || when < p.when) { break; } if (needWake && p.isAsynchronous()) { needWake = false; } } msg.next = p; // invariant: p == prev.next prev.next = msg; } // We can assume mPtr != 0 because mQuitting is false. if (needWake) { nativeWake(mPtr); } } return true;}

1.在消息对象msg中包含一个when变量,保存的就是该消息应执行的时间(刚才转化后的绝对值时间)。 2.mMessagesMessageQueue类中的一个全局变量,指向的是消息队列中的头节点。因此我们看到判断逻辑前,先将mMessages引用赋值给了变量p。 3.接下来,若消息队列中没有任何消息(p=null),或绝对时间为0,亦或这条消息执行时间在当前队列第一条消息之前,那么将这条消息添加到队列开始的位置; 不满足这些条件时,用一个很典型的链表插入算法,将消息按时间顺序添加到列表中。


发表评论 共有条评论
用户名: 密码:
验证码: 匿名发表