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

Android的Handler在发送消息或设定延迟后,如何处理消息队列?本文深入探讨Handler源码,揭示从设定延迟到消息入队列的过程。首先,延迟时间通过与系统启动时间相加得到绝对值,然后消息在Looper中按时间顺序插入队列,确保正确执行顺序。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

在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,亦或这条消息执行时间在当前队列第一条消息之前,那么将这条消息添加到队列开始的位置;
不满足这些条件时,用一个很典型的链表插入算法,将消息按时间顺序添加到列表中。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值