Android中HandlerThread的使用及源码解析

2024-02-19 19:18

本文主要是介绍Android中HandlerThread的使用及源码解析,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

关于Hanlder的基本使用可以参见博文《Android中Handler的使用》,如果想了解Handler、Looper、Thread等的相互关系以及内部实现原理可以参见博文《深入源码解析Android中的Handler,Message,MessageQueue,Looper》。

Android中的API中对HandlerThread的描述是:
Handy class for starting a new thread that has a looper. The looper can then be used to create handler classes. Note that start() must still be called.
意思是HandlerThread类可以很方便地创建一个带有looper的新线程。该looper可以被用来创建hanlder对象。需要注意的是start方法必须要调用。

先抛开HanlderThread,我们不用这个类看看怎么使用Handler、Thread、Looper。

我们可以通过Looper.myLooper()方法得到当前线程所关联的looper对象。在创建一个新线程的时候,初始情况下新线程是没有关联looper以及对应的消息队列MessageQueue的,对外表现出来就是在该新线程中调用Looper.myLooper()返回null。如果我们没有意识到这一点,那么我们在新线程中使用Handler肯能就会遇到问题。

假设为了在新线程中使用使用Handler,我们可能会写出如下的代码:

class TestThread extends Thread {public Handler mHandler;public void run() {mHandler = new Handler() {public void handleMessage(Message msg) {// process incoming messages here}};}}

但是在实际执行的时候会发现当运行到mHandler = new Handler()这一句时就会抛出异常:
Can’t create handler inside thread that has not called Looper.prepare()
之所以会抛出异常,可参见Handler构造函数的源码。

抛出异常的原因是: 我们在构造函数中没有传递Looper,这样Hanlder在构造函数中就使用默认的looper,默认的looper是通过调用Looper.myLooper()得来的。当我们调用了Looper.prepare()之后,我们就会将looper关联到当前线程中。因此只有在调用了Looper.prepare()这个方法之后,Looper.myLooper()才能得到looper对象。所以这里提示我们要先调用Looper.prepare()方法才行。

为了能在新线程中正常创建使用Handler,我们将代码改成如下所示:

class LooperThread extends Thread {public Handler mHandler;public void run() {Looper.prepare();mHandler = new Handler() {public void handleMessage(Message msg) {//此处处理消息}};Looper.loop();}
}

我们在新线程的run方法中,首先调用了Looper.prepare()方法,这样就将looper对象关联到当前线程中了,然后执行new Handler(),在Hanlder的构造函数内部会调用Looper.myLooper()得到当前线程所关联的looper对象。在创建完Hanlder对象之后,我们需要调用Looper.loop()方法让消息队列循环起来。

通过上面的代码我们就可以在一个新线程中创建并使用Handler对象了,但是问题是每次这么写感觉很罗嗦,不方便。为了让能开发者更方便地在新线程中创建并使用Handler,Android提供了HandlerThread这个类,HandlerThread是继承自Thread类的。

使用HandlerThread的示例代码如下:

HandlerThread handlerThread = new HandlerThread("TestHandlerThread");
handlerThread.start();
Handler handler = new Handler(handlerThread.getLooper()) {public void handleMessage(Message msg) {//此处处理消息};
};

我们创建了HandlerThread之后需要先调用其start方法,调用start方法之后,run方法就会在HanlderThread线程中执行了。

HandlerThread这个类的run方法的源码如下所示:

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

我们可以看到,在该run方法中也是先调用了Looper.prepare()方法,然后通过Looper.myLooper()方法得到该线程所关联的looper对象,最后会调用Looper.loop()方法让消息队列循环起来。由此可以看出,HandlerThread的run方法主要就是将我们上面给出的正常情况下在新线程中创建Handler的代码做了一些封装而已。 在创建HandlerThread对象并调用其start方法之后,该HandlerThread线程就已经关联了looper对象(通过Looper.prepare()方法关联),并且该线程内部的消息队列循环了起来(通过Looper.loop()方法)。 最后我们只需要在创建Handler对象的时候通过handlerThread.getLooper()将handlerThread线程所关联的looper对象传递给Handler的构造函数即可。 正如本文开头API对HandlerThread所解释的那样: HandlerThread类可以很方便地创建一个带有looper的新线程。该looper可以被用来创建hanlder对象。需要注意的是start方法必须要调用。

HandlerThread使用起来之所以感觉方便,是因为HandlerThread这个类在run方法内部对Looper做了一些工作(调用Looper.prepare()和Looper.loop()方法),这样我们开发者在使用的时候就不需要太多的与Looper打交道了,从而提升开发的便利性。HandlerThread并不是很高深的,只是对我们常见的开发流程做了封装而已,因此我们不用HandlerThread而自己去实现也是可以的,具体用不用HandlerThread根据自己的喜好而定。

这篇关于Android中HandlerThread的使用及源码解析的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/725679

相关文章

Spring Boot整合Redis注解实现增删改查功能(Redis注解使用)

《SpringBoot整合Redis注解实现增删改查功能(Redis注解使用)》文章介绍了如何使用SpringBoot整合Redis注解实现增删改查功能,包括配置、实体类、Repository、Se... 目录配置Redis连接定义实体类创建Repository接口增删改查操作示例插入数据查询数据删除数据更

使用python生成固定格式序号的方法详解

《使用python生成固定格式序号的方法详解》这篇文章主要为大家详细介绍了如何使用python生成固定格式序号,文中的示例代码讲解详细,具有一定的借鉴价值,有需要的小伙伴可以参考一下... 目录生成结果验证完整生成代码扩展说明1. 保存到文本文件2. 转换为jsON格式3. 处理特殊序号格式(如带圈数字)4

Java使用Swing生成一个最大公约数计算器

《Java使用Swing生成一个最大公约数计算器》这篇文章主要为大家详细介绍了Java使用Swing生成一个最大公约数计算器的相关知识,文中的示例代码讲解详细,感兴趣的小伙伴可以了解一下... 目录第一步:利用欧几里得算法计算最大公约数欧几里得算法的证明情形 1:b=0情形 2:b>0完成相关代码第二步:加

Java中流式并行操作parallelStream的原理和使用方法

《Java中流式并行操作parallelStream的原理和使用方法》本文详细介绍了Java中的并行流(parallelStream)的原理、正确使用方法以及在实际业务中的应用案例,并指出在使用并行流... 目录Java中流式并行操作parallelStream0. 问题的产生1. 什么是parallelS

Linux join命令的使用及说明

《Linuxjoin命令的使用及说明》`join`命令用于在Linux中按字段将两个文件进行连接,类似于SQL的JOIN,它需要两个文件按用于匹配的字段排序,并且第一个文件的换行符必须是LF,`jo... 目录一. 基本语法二. 数据准备三. 指定文件的连接key四.-a输出指定文件的所有行五.-o指定输出

Java中Redisson 的原理深度解析

《Java中Redisson的原理深度解析》Redisson是一个高性能的Redis客户端,它通过将Redis数据结构映射为Java对象和分布式对象,实现了在Java应用中方便地使用Redis,本文... 目录前言一、核心设计理念二、核心架构与通信层1. 基于 Netty 的异步非阻塞通信2. 编解码器三、

Linux jq命令的使用解读

《Linuxjq命令的使用解读》jq是一个强大的命令行工具,用于处理JSON数据,它可以用来查看、过滤、修改、格式化JSON数据,通过使用各种选项和过滤器,可以实现复杂的JSON处理任务... 目录一. 简介二. 选项2.1.2.2-c2.3-r2.4-R三. 字段提取3.1 普通字段3.2 数组字段四.

Linux kill正在执行的后台任务 kill进程组使用详解

《Linuxkill正在执行的后台任务kill进程组使用详解》文章介绍了两个脚本的功能和区别,以及执行这些脚本时遇到的进程管理问题,通过查看进程树、使用`kill`命令和`lsof`命令,分析了子... 目录零. 用到的命令一. 待执行的脚本二. 执行含子进程的脚本,并kill2.1 进程查看2.2 遇到的

Java HashMap的底层实现原理深度解析

《JavaHashMap的底层实现原理深度解析》HashMap基于数组+链表+红黑树结构,通过哈希算法和扩容机制优化性能,负载因子与树化阈值平衡效率,是Java开发必备的高效数据结构,本文给大家介绍... 目录一、概述:HashMap的宏观结构二、核心数据结构解析1. 数组(桶数组)2. 链表节点(Node

详解SpringBoot+Ehcache使用示例

《详解SpringBoot+Ehcache使用示例》本文介绍了SpringBoot中配置Ehcache、自定义get/set方式,并实际使用缓存的过程,文中通过示例代码介绍的非常详细,对大家的学习或者... 目录摘要概念内存与磁盘持久化存储:配置灵活性:编码示例引入依赖:配置ehcache.XML文件:配置