Handler 中的 handleMessage 所在线程是由什么决定的?

简介: 大多数情况下,handleMessage所在线程和 handler 初始化所在的线程相同,但 handler 初始化的时候可以传入一个 Looper 对象,此时handleMessage所在线程和参数looper所在线程相同。

大多数情况下,handleMessage所在线程和 handler 初始化所在的线程相同,但 handler 初始化的时候可以传入一个 Looper 对象,此时handleMessage所在线程和参数looper所在线程相同。

1. 含参构造public Handler(Looper looper)

class MainActivity : AppCompatActivity() {
    var handler: Handler? = null
    var looper: Looper? = null
    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        setContentView(R.layout.activity_main)
        looper = Looper.getMainLooper()
        val thread = object : Thread() {
            override fun run() {
                super.run()
                Log.e("abc", "--- Runnable:threadName ---" + Thread.currentThread().name)
                handler = object : Handler(looper) {
                    override fun handleMessage(msg: Message?) {
                        super.handleMessage(msg)
                        Log.e("abc","--- handleMessage:threadName ---" + Thread.currentThread().name
                        )
                    }
                }
            }
        }
        thread.start()
        myBtn.setOnClickListener {
            val msg = Message()
            handler!!.sendMessage(msg)
        }
    }
}

// log 打印情况
--- Runnable:threadName ---Thread-2
--- handleMessage:threadName ---main

从 log 中可以看到 handler 初始化所在线程在 Thread-2,而handleMessage所在的线程是主线程main.

2. 无参构造

如果使用无参的 Handler 初始化构造,需要手动调用Looper.prepare()Looper.loop()

val thread = object : Thread() {
            override fun run() {
                super.run()
                Log.e("abc", "--- Runnable:threadName ---" + Thread.currentThread().name)
                Looper.prepare()
                handler = object : Handler() {
                    override fun handleMessage(msg: Message?) {
                        super.handleMessage(msg)
                        Log.e(
                            "abc", "--- handleMessage:threadName ---" + Thread.currentThread().name
                        )
                    }
                }
                Looper.loop()
            }
        }

// log 打印情况
--- Runnable:threadName ---Thread-2
--- handleMessage:threadName ---Thread-2

不手动调用Looper.prepare()会抛出异常:

java.lang.RuntimeException: Can't create handler inside thread Thread[Thread-2,5,main] that has not called Looper.prepare()

主线程中使用 Handler
大多数时候我们不会在子线程中初始化和使用 handler,而是在主线程中使用,此时不需要prepare()loop(),因为主线程中自带一个 Looper(通过Looper.getMainLooper()可以获取)

3. 一个线程可以有多少个 Looper?Handler 和 Looper 之间如何关联?

3.1 一个线程可以有多少个 Looper

查看Looper.prepare()源码:

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

继续查看sThreadLocal.set(new Looper(quitAllowed))

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

Threadlocal 是一个线程内部的存储类,可以在指定线程内存储数据,数据存储以后,只有指定线程可以得到存储数据。在这里 ThreadLocal 的作用是保证了每个线程都有各自的 Looper,就是说一个线程只能有一个 Looper,关于 Threadlocal,可以看看这篇文章 Threadlocal

接下来看看创建 Looper 实例的方法new Looper(quitAllowed)

final MessageQueue mQueue;
final Thread mThread;
private Looper(boolean quitAllowed) {
        mQueue = new MessageQueue(quitAllowed);
        mThread = Thread.currentThread();

在构造方法里,初始化 了MessageQueue 和代表当前线程的属性 mThread.

调用Looper.prepare()其实就是利用 ThreadLocal 为当前的线程创建了一个独立的 Looper,这其中包含了一个消息队列

3.2 Handler 和 Looper 之间如何关联

一个线程只能有一个 Looper,但一个线程中可以创建多个 Handler,那么一个 Looper 怎么和多个 Handler 对应呢?查看源码可知,post(Runnable r)postDelayed(Runnable r, long delayMillis)postAtTime(Runnable r, long uptimeMillis)sendMessage最终调用的都是enqueueMessage方法:

private boolean enqueueMessage(MessageQueue queue, Message msg, long uptimeMillis) {
        msg.target = this;
        if (mAsynchronous) {
            msg.setAsynchronous(true);
        }
        return queue.enqueueMessage(msg, uptimeMillis);
    }

msg.target = this这里就是将当前的 Handler 赋值给 Message 对象的 target 属性,这样在处理消息的时候通过msg.target就可以区分开不同的 Handler 了。

相关文章
|
存储 消息中间件 Android开发
Handler切换线程原理解析
写在前面:本文的目的是想将Handler、Looper和Thread之间绑定的原理讲明白,如果没讲明白,也希望能给关于Handler的学习留个印象。 Android中的多线程间交互离不开Handler,开发中最常见的操作是在子线程中执行耗时操作,在主线程中更新UI,这其中就涉及到了Handler的线程切换操作。
|
消息中间件 存储 Java
【Android 异步操作】手写 Handler ( Message 消息 | ThreadLocal 线程本地变量 | Looper 中的消息队列 MessageQueue )
【Android 异步操作】手写 Handler ( Message 消息 | ThreadLocal 线程本地变量 | Looper 中的消息队列 MessageQueue )
172 0
|
机器学习/深度学习 消息中间件 Android开发
Android多线程源码详解一:handler、looper、message、messageQueue
之前面试,面试官问到多线程通讯,巴拉巴拉说了些基础实现后面试官问handlerThread的底层实现,就卡住了。所以把Android多线程的知识点复习整理一下,写出来加深印象。 Android多线程通讯的核心是handler、looper、message、messageQueue,这篇文章就先记录下这套系统的源码要点,具体的实现方法下一篇文章再写。
|
Android开发
Android多线程源码详解一:handler、looper、message、messageQueue
Android多线程源码详解一:handler、looper、message、messageQueue之前面试,面试官问到多线程通讯,巴拉巴拉说了些基础实现后面试官问handlerThread的底层实现,就卡住了。
1558 0
|
Java Android开发
Android非UI主线程中,若干普通Java线程使用Handler发送接收消息
Android非UI主线程中,若干普通Java线程使用Handler发送接收消息 线程1和线程2均为普通Java线程,在Android中创建,然后在这两个普通Java线程中使用Handler发送和接收消息。
1233 0
|
消息中间件 Android开发
|
Android开发 消息中间件