当前位置:网站首页>Handler 源码解析

Handler 源码解析

2022-08-03 19:03:00 故、梦

Handler 用于异步消息的处理,是 Android 面试中的常客。这篇博客用于详细说明 Handler 源码解析

同步和异步

在聊 Handler 之前,我们先了解一下同步和异步。

  • 同步:一个进程在执行某个请求时,如果该请求需要一段时间才能返回信息,那么这个进程会一直等待,直到信息返回才继续执行。
  • 异步:进程在执行某个程序时,如果请求需要一段时间才能返回信息,此时进程不需要等待,可以继续执行下面的操作,当有信息返回的时候会通知进程处理。

例子:你和你女友约会,你们约定八点见面。已知你女友七点五十才醒,化妆要化 2 小时。

同步:你按时到了,然后一直搁那等,直到女友下来。

异步:你和女友说,你准备好了给我发个消息,然后跑到网吧打游戏,直到女友化好妆下来。

Handler 背后的生产者-消费者模型

生产者-消费者模型:生产者和消费者共用同一个存储空间,生产者往存储空间中放数据,消费者从存储空间中取数据。生产者和消费者互相不持有,当没有数据时,消费者线程挂起,有数据时将消费者线程唤醒;当存储空间满时,生产者线程挂起,当空间不满时「消费者拿走了数据」唤醒生产者线程。

在这里插入图片描述

Handler 的异步处理流程

Android 的异步消息处理机制主要由四个部分组成:Message、Handler、MessageQueue 和 Looper。

Message

Message 是在线程之间传递的消息,它可以在内部携带少量的信息,用于在不同的线程之间传递数据。除了 what 字段,还可以使用 arg1 和 arg2 字段来携带一些整型数据,使用 obj 字段携带一个 object 对象。

Handler

Handler 是处理者的意思,它用于发送和处理消息。发送消息使用 handler 的 sendMessage() 方法,post 方法等,发出的消息经过一系列地辗转处理后,会到达 Handler 的 handleMessage() 方法中。

MessageQueue

MessageQueue 是消息队列的意思,它用于存储所有 Handler 发送的消息。直到消息被处理,它会一直存在消息队列中。「每个线程中只会有一个 MessageQueue 对象」

Looper

Looper 是 MessageQueue 的管家,当调用了 Looper 的 loop() 方法后,就会进入一个无限循环中,每当发现 MessageQueue 中存在一条消息,就把它取出,并传递到 Handler 的handleMessage() 方法中。「每个线程只有一个 Looper 对象」

异步处理流程

  • 首先,在主线程创建一个 Handler 对象,并重写handleMessage() 方法
  • 子线程需要处理 UI 操作时,就创建一个 Message 对象,并通过 Handler 将该消息发送出去
  • 发出的消息会被添加到 MessageQueue 中,等待被处理
  • Looper 一直看 MessageQueue 中有没有消息,有的话就分发回 Handler 的 handleMessage()中。在 Hanlder 的构造函数中,我们传入了 Looper.getMainLooper(),所以handleMessage() 方法会在主线程中运行。

在这里插入图片描述

Handler 相关类解析

创建 Looper

Looper 的创建使用的是 Looper.prepare() 方法

App 的启动流程:launcher(app) :zygote → jvm → ActivityThread.main

主线程 Looper 的创建就在 ActivityThread 的 Main 方法中

public static void main(String[] args) {
     	
  	Trace.traceBegin(Trace.TRACE_TAG_ACTIVITY_MANAGER, "ActivityThreadMain");
  	//其他代码省略... 
		Looper.prepareMainLooper(); //初始化Looper以及MessageQueue
    ActivityThread thread = new ActivityThread();
    thread.attach(false);
    if (sMainThreadHandler == null) {
    
        sMainThreadHandler = thread.getHandler();
}
    if (false) {
    
        Looper.myLooper().setMessageLogging(new LogPrinter(Log.DEBUG, "ActivityThread"));
}
        // End of event ActivityThreadMain.
Trace.traceEnd(Trace.TRACE_TAG_ACTIVITY_MANAGER); Looper.loop(); //开始轮循操作
        throw new RuntimeException("Main thread loop unexpectedly exited");
    }

代码中调用的是 Looper.prepareMainLooper()方法,我们来细看一下这个方法

 @Deprecated
    public static void prepareMainLooper() {
    
        prepare(false); // 传入 false 表示不可以 quit
        synchronized (Looper.class) {
    
            if (sMainLooper != null) {
    
                throw new IllegalStateException("The main Looper has already been prepared.");
            }
            sMainLooper = myLooper();
        }
    }

它调用了一个 prepare() 方法,用于创建消息队列,然后用同步的方式创建一个主线程 Looper。prepare() 的参数为 quitAllowed,它用于标识创建的消息队列是否可以销毁

private static void prepare(boolean quitAllowed) {
    
        if (sThreadLocal.get() != null) {
     // 如果不为空,表示 Looper 存在
            throw new RuntimeException("Only one Looper may be created per thread");
        }
        sThreadLocal.set(new Looper(quitAllowed)); // 创建一个 Looper
}

创建 MessageQueue 并绑定当前线程

查看 Looper 的构造函数,我们会发现它做了两件事:(1)创建 MessageQueue (2)与当前线程绑定

 private Looper(boolean quitAllowed) {
    
        mQueue = new MessageQueue(quitAllowed); // 创建消息队列
        mThread = Thread.currentThread();  // 绑定当前线程
    }

我们继续查看 MessageQueue 的构造方法

MessageQueue(boolean quitAllowed) {
    
        mQuitAllowed = quitAllowed; 	// 消息队列是否可以被销毁,主线程不可被销毁
        mPtr = nativeInit();
}

需要注意的是主线程的 MessageQueue 不可以被销毁,销毁 MessageQueue 可以调用 MessageQueue的 quit() 方法,实际上是往消息队列中放入了一个空的 Message。

Looper.loop()

同样在 ActivityThread.main 方法中,调用 Looper.loop()方法后开始轮询

public static void loop() {
    
    final Looper me = myLooper();//里面调用了sThreadLocal.get()获得刚才创建的Looper对象 if (me == null) {
    
    throw new RuntimeException ("No Looper; Looper.prepare() wasn't called on this }//如果Looper为空则会抛出异常
            thread .");
    final MessageQueue queue = me.mQueue;

    Binder.clearCallingIdentity();
    final long ident = Binder.clearCallingIdentity();
    for (;;) {
    //这是一个死循环,从消息队列不断的取消息
        Message msg = queue.next (); // might block if (msg == null) {
    
        //由于刚创建MessageQueue就开始轮询,队列里是没有消息的,等到Handler sendMessage enqueueMessage后 队列里才有消息
				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
        );
    }
    msg.target.dispatchMessage(msg);//msg.target就是绑定的Handler,详见后面Message的部 分,Handler开始
//后面代码省略..... msg.recycleUnchecked();
}}

创建 Handler

Handler 最常见的创建方式如下:

Handler handler=new Handler(){
     //这里调用的是 Handler(null,false) 构造方法
    @Override
    public void handleMessage(Message msg) {
    
        super.handleMessage(msg);
} };

Handler 的构造函数如下

 public Handler(@Nullable Callback callback, boolean async) {
    
   			// 前面省略
        mLooper = Looper.myLooper(); // 获取 Looper
        if (mLooper == null) {
    
            throw new RuntimeException(
                "Can't create handler inside thread " + Thread.currentThread()
                        + " that has not called Looper.prepare()");
        }
        mQueue = mLooper.mQueue;
        mCallback = callback; // 初始化回调
        mAsynchronous = async; // 该 Handler 是否为异步
    }

创建 Message

创建 Message 可以直接 new,但更好的方式是使用 Message.obtain()。它可以检查是否有可以复用的 Message,避免过多的创建和销毁 Message,进而达到优化内存和性能的目的

public static Message obtain(Handler h) {
    
      Message m = obtain(); // 调用重载的 obtain 方法
      m.target = h; // 绑定 Handler

      return m;
 }
public static Message obtain() {
    
      synchronized (sPoolSync) {
     // sPoolSync 是一个 object 对象,用来同步保证线程安全
          if (sPool != null) {
     // sPool 是 handler.dispatchMessage 后 通过 recycleUncheck 回收用以复用的 Message
              Message m = sPool;
              sPool = m.next;
              m.next = null;
              m.flags = 0; // clear in-use flag
              sPoolSize--;
              return m;
          }
      }
      return new Message();
}

Handler 发送和处理消息

发送消息通过 Handler 的 sendMessage() 方法,将 Message 送到 MessageQueue 中,Looper 从 MessageQueue 中取出消息,然后派遣给 Handler,Handler 调用 handleMessage()方法处理。

我们可以把 Handler 机制看作是一个传送带:

  • MessageQueue 是履带
  • Thread 是动力系统
  • Looper.loop() 是传送带的开关
  • Message 是我们的货物

我们可以看出 Message 在 MessageQueue 中是有序的

Handler 的难点问题

1.线程同步问题

Handler 的使用范围非常广,小到 UI 处理,大到整个 app 都线程间协调,都需要 Handler 。因此,Handler 中的线程安全尤为重要,在 Handler 中线程安全是怎么实现的呢?

Handler 中最主要的类是 MessageQueue ,它是所有消息的存储仓库,我们需要在这个类中做好消息管理。消息管理包括两步:(1)消息入库(enqueueMessage)(2)消息出库 (next)

enqueueMessage()的源码如下:

boolean enqueueMessage(Message msg, long when) {
    
        if (msg.target == null) {
     	// Message 的 target 指向 Handler
            throw new IllegalArgumentException("Message must have a target.");
        }

        synchronized (this) {
     // 加锁
            if (msg.isInUse()) {
    
                throw new IllegalStateException(msg + " This message is already in use.");
            }

            if (mQuitting) {
    
                IllegalStateException e = new IllegalStateException(
                        msg.target + " sending message to a Handler on a dead thread");
                Log.w(TAG, e.getMessage(), e);
                msg.recycle();
                return false;
            }

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

通过内置锁 synchronized 来实现enqueueMessage() 的线程安全。前面说到,一个线程对应唯一一个 Handler 对象,而 Looper 中又有唯一一个 MessageQueue,所以当子线程往主线程发送消息时,主线程一次只会处理一个消息,其他的都要等待,通过这种方式来保证消息队列不出现混乱。

接下来,我们看 MessageQueue 的 next()函数

Message next() {
    
    ....
    for (;;) {
    
        ....
        nativePollOnce(ptr, nextPollTimeoutMillis);
        synchronized (this) {
    
            // Try to retrieve the next message. Return if found.
                   ...
                    return msg;
}
} else {
    
                // No more messages.
                nextPollTimeoutMillis = -1;
            }
... }//synchronized 结束之处
        // Run the idle handlers.
        // We only ever reach this code block during the first iteration.
        for (int i = 0; i < pendingIdleHandlerCount; i++) {
    
            final IdleHandler idler = mPendingIdleHandlers[i];
            mPendingIdleHandlers[i] = null; // release the reference to the handler
            boolean keep = false;
            try {
    
                keep = idler.queueIdle();
            } catch (Throwable t) {
    
                Log.wtf(TAG, "IdleHandler threw exception", t);
            }
            if (!keep) {
    
                synchronized (this) {
    
                    mIdleHandlers.remove(idler);
                }
} }
        // Reset the idle handler count to 0 so we do not run them again.
        pendingIdleHandlerCount = 0;
        // While calling an idle handler, a new message could have been delivered
        // so go back and look again for a pending message without waiting.
        nextPollTimeoutMillis = 0;
} }

看到 next() 函数,我们可能会存在疑问:我们从线程中取消息,每次都是从头部拿,那加锁还有什么意义呢?实际上,我们加锁使用的是 synchronized(this),它作用的范围是整个 MessageQueue,它能保证 next()函数和 enqueueMessage()函数实现互斥,保证多线程访问时 MessageQueue 有序进行。

2. 同步屏障

消息队列中,消息都放在同一个 MessageQueue 中,取消息的时候是互斥的,且只能从头部取消息,这也就是说消息的执行按照消息队列的顺序。假如我们现在有一个紧急消息需要处理,该怎么办呢?在 Android 中,我们需要给紧急执行的消息一个绿色屏障,也就是所谓的同步屏障。

同步屏障顾名思义就是**阻碍同步消息,只让异步消息通过。**同步屏障的实现借助于 MessageQueue.postSynBarrier()方法

 public int postSyncBarrier() {
    
        return postSyncBarrier(SystemClock.uptimeMillis());
    }

    private int postSyncBarrier(long when) {
    
        // Enqueue a new sync barrier token.
        // We don't need to wake the queue because the purpose of a barrier is to stall it.
        synchronized (this) {
    
            final int token = mNextBarrierToken++;
            final Message msg = Message.obtain(); 	// 从消息池中取 Message
            msg.markInUse(); 	// 启用 Message
            msg.when = when;
            msg.arg1 = token;

            Message prev = null;
            Message p = mMessages;// 表头节点
            if (when != 0) {
    
                while (p != null && p.when <= when) {
    
                    prev = p;
                    p = p.next;
                }
            }
          // 如果 prev 不是 null,将 msg 按照时间顺序插入到消息队列的合适位置
            if (prev != null) {
     // invariant: p == prev.next
                msg.next = p;
                prev.next = msg;
            } else {
    
                msg.next = p;
                mMessages = msg;
            }
            return token;
        }
    }

Message 对象在初始化时并没有给 target 赋值,因此设置同步屏障后,就存在有 target == null 的消息。当我们调用 MessageQueue.next()函数中,如果碰到了 target == null 的消息,就会过滤掉后方的同步消息,转而处理异步消息「同步和异步是指 Handler 的类型」。

image-20220712200242621

也就是说只要有同步屏障,就只会处理异步消息。那么什么时候同步消息可以执行呢?这就需要调用 removeSynBarrier() 方法来移除这个屏障。举个例子,现在有一批病人需紧急送往医院,为了保证速度,交警在路上设卡,不让救护车以外的车辆通过,当所有救护车通过后,就可以取消卡外,让普通车辆通过。

同步屏障的应用场景

开发中,一个很常见的同步屏障使用场景就是 Android 系统更新 UI ,更新 UI 为异步消息,需要紧急处理。「以 60 HZ 刷新率为例,大概 16.67 ms 需要刷新一次屏幕」

在 ViewRootImpl 的 scheduleTraversals() 方法中

void scheduleTraversals() {
    
        if (!mTraversalScheduled) {
    
            mTraversalScheduled = true;
          // 开启同步屏障
            mTraversalBarrier = mHandler.getLooper().getQueue().postSyncBarrier();
          // 发送异步消息
            mChoreographer.postCallback(
                    Choreographer.CALLBACK_TRAVERSAL, mTraversalRunnable, null);
            notifyRendererOfFramePending();
            pokeDrawLockIfNeeded();
        }
    }

postCallback() 方法最终会走到 postCallbackDelayedInternal()

 private void postCallbackDelayedInternal(int callbackType,
            Object action, Object token, long delayMillis) {
    
        if (DEBUG_FRAMES) {
    
            Log.d(TAG, "PostCallback: type=" + callbackType
                    + ", action=" + action + ", token=" + token
                    + ", delayMillis=" + delayMillis);
        }

        synchronized (mLock) {
    
            final long now = SystemClock.uptimeMillis();
            final long dueTime = now + delayMillis;
            mCallbackQueues[callbackType].addCallbackLocked(dueTime, action, token);

            if (dueTime <= now) {
    
                scheduleFrameLocked(now);
            } else {
    
                Message msg = mHandler.obtainMessage(MSG_DO_SCHEDULE_CALLBACK, action);
                msg.arg1 = callbackType;
                msg.setAsynchronous(true); // 创建异步消息
                mHandler.sendMessageAtTime(msg, dueTime); // 发送异步消息
            }
        }
    }

这样就开启了同步屏障,并发送了异步消息,由于 UI 更新相关消息的优先级是最高的,所以它会很快得到执行。移除同步屏障就需要调用 ViewRootImpl.unscheduleTraversals()

void unscheduleTraversals() {
    
        if (mTraversalScheduled) {
    
            mTraversalScheduled = false;
          	// 移除同步屏障
            mHandler.getLooper().getQueue().removeSyncBarrier(mTraversalBarrier); 
            mChoreographer.removeCallbacks(
                    Choreographer.CALLBACK_TRAVERSAL, mTraversalRunnable, null);
        }
    }

在子线程处理消息

在子线程处理消息,可以使用 HandlerThread。它是 Thread 的子类,它严格意义上是一个线程,它在内部帮哦我们创建好了 Looper,它具有以下特点:

  • 方便初始化和获得线程 Looper
  • 保证了线程安全

因为是在线程中创建,我们可以查看 HandlerThread 的 run()方法

 public void run() {
    
        mTid = Process.myTid();
        Looper.prepare();
        synchronized (this) {
    
            mLooper = Looper.myLooper();
            notifyAll();
        }
        Process.setThreadPriority(mPriority);
        onLooperPrepared();
        Looper.loop();
        mTid = -1;
    }

Handler 的常见面试题

  1. 一个线程有几个 Handler ?

    可以有任意个,可以在任何时候 new() 出来。流水线只有一条,但工人可以有很多。

  2. 一个线程有几个 Looper?如何保证?

​ 只有一个 Looper,使用 ThreadLocal.get() 方法来保证。

  1. Handler 内存泄漏的原因?为什么其他内部类没有出现过这个问题?

Message.target 指向具体的 Handler,也就是 Message 持有了Handler。Handler 可以调用 Activity 的方法,表明 Handler 持有 Activity 对象。如果我们设置了一个 20 分钟后执行的 Message,那么在这 20 分钟内,Message 会一直持有 Handler,Handler 会一直持有 Activity ,进而产生内存泄露。

  1. 为何主线程可以 new Handler()?如果要在子线程中 new Handler()需要怎么做?

​ 因为程序启动时,主线程会调用 Looper.prepareMainLooper()。这样主线程中就有了 Looper,就可以 new Handler,前面说过 Looper 就好比流水线,Handler 是工人,流水线都没有,你招工人吃空饷吗?所以子线程想要 new Handler 就先得有流水线 Looper

Looper.prepare()
Looper.loop() // 流水线开机了才可以让工人工作嘛
  1. 子线程中维护的 Looper,消息队列中无消息的时候处理方案是什么?有什么用?

​ 队列中无消息的时候,可以调用 Looper.quit() 方法停止 loop()quit() 方法会清空消息队列中的所有消息,然后传入一个为空的 Message,进而让 loop()停止。

  1. Looper 死循环为什么不会导致线程卡死

ANR 问题的出现是消息超时响应,而 Looper 死循环是取消息的,所以并不是一个问题,偷换了概念。此外,ANR 的触发,也通过 Looper 取用弹出 ANR 消息。

原网站

版权声明
本文为[故、梦]所创,转载请带上原文链接,感谢
https://blog.csdn.net/jiaweilovemingming/article/details/126112239