Android线程间通信 Handler使用详解
作者:Quincy_Ye 发布时间:2022-10-16 09:22:33
前言
Handler,可谓是面试题中的一个霸主了。在我《面试回忆录》中,几乎没有哪家公司,在面试的时候是不问这个问题的。简单一点,问问使用流程,内存泄漏等问题。复杂一点,纠其源码细节和底层 epoll 机制来盘你。所以其重要性,不言而喻了吧。
那么今天让我们来揭开 Handler 的神秘面纱。为了读者轻松易读不烧脑,本系列文章按照使用篇,源码篇,面试篇来分篇浅析。
01、定义
在了解如何使用前,我们先来看看什么是 Handler ?
A Handler allows you to send and process and Runnable objects associated with a thread's .
可以用 Handler 发送并且处理与线程有关的 Runnable 对象。
这听起来,可能还有点迷惑。那么讲一个常见的场景来引出 Handler 吧。你接到了一个需求:有一个文件列表,点击文件后,开始下载该文件,下载完成后,在列表中对应文件上加上已下载的标识。
下载是一个耗时的动作,为了不引起 ANR(Application not response),我们通常需要通过子线程去完成这任务。但通常情况下(有特殊情况),我们又不能在非主线程中去更新 UI,所以就需要进行线程间通信。而 Handler 便发挥其作用了——进行线程间通信。
02、使用
第一步、创建
在创建 Handler 的时候,需要有注意一下是在主线程还是子线程。
//主线程
private val mHandler: Handler = object : Handler(Looper.getMainLooper()) {
override fun handleMessage(msg: Message) {
when (msg.what) {
1 -> {}
}
}
}
//子线程
Thread {
Looper.prepare()
val handler = object : Handler() {
override fun handleMessage(msg: Message) {
when (msg.what) {
1 -> {}
}
}
}
handler.sendEmptyMessage(1)
Looper.loop()
}.start()
在主线程创建我们无需调用 Looper.prepare()
和 Looper.loop()
;
在子线程需要手动调用者两个方法。至于为什么主要是主线的这两个方法,在应用入口 main()
中系统帮我们完成了,具体见后续原理篇。
第二步、发送消息
发送消息有两大类方式(此处忽略定时和延迟发送)
第一种是 post(Runnable)
// 未简化写法,为了更好理解
mHandler.post(object :Runnable{
override fun run() {
// TODO 具体业务逻辑
}
})
第二种是 sendMessage(Message)
val msg = Message()
msg.what = 1
msg.obj = "Quincy"
mHandler.sendMessage(msg)
但其实两者本质上是没有区别的。因为上面两种方式最后都调用了 sendMessageDelayed()
,只是源码上帮我们把 Runnable 包装成了一个 Message。具体看下一篇,源码篇。
第三步、处理消息
处理消息,并非简单地在 handleMessage()
中处理即可,这还是要分情况的。为了方便分析,我们先浅浅窥探一下处理消息的源码
public void dispatchMessage(Message msg) {
if (msg.callback != null) {
handleCallback(msg); // 注释1
} else {
if (mCallback != null) {
if (mCallback.handleMessage(msg)) {
return; //注释2
}
}
handleMessage(msg); //注释3
}
}
注意上面三点注释,它们之间的关系。如果 Message.callback
不为空,即通过 post()
发送消息的,则调用 handleCallback(Message)
private static void handleCallback(Message message) {
message.callback.run();
}
否则,如果 mCallback 不为空且拦截了消息,则不再调用 handleMessage()
,
private val handler = object : Handler(object : Callback {
override fun handleMessage(msg: Message): Boolean {
if (msg.what == 1) {
return true//不会调用注释3
}
return false//会调用注释3
}
}) {
override fun handleMessage(msg: Message) {
when (msg.what) {
1 -> {}
}
}
}
否则将回调。
private val mHandler: Handler = object : Handler() {
override fun handleMessage(msg: Message) {
when (msg.what) {
1 -> {}
}
}
}
03、结语
一个线程中有多个 Handler ,但只有 1 个 Looper 和 1个 MessageQueue。哪在分发消息的时候,怎么知道发个哪一个 Handler 呢?
Looper 中有多少个
for( ; ; )
,分别的作用是什么呢?消息被处理后,Message 是被怎么处理的呢?
Handler 为什么会引起内存泄漏?
Handler 中有 Looper 的死循环,为什么没有卡死呢?(我认为就是要卡“死”,正是因为它我们的程序才没有退出)
来源:https://juejin.cn/post/7150287876276617253
猜你喜欢
- 字符, 字节与字符串字符与字符串字符串内部包含一个字符数组,String 可以和 char[] 相互转换.字符数组变为字符串:public
- 前言上一篇通过clusterservice对cluster做了一个简单的概述, 应该能够给大家一个初步认识。本篇将对cluster的代码组成
- 场景:有一个喜欢吃饺子,他有三种不同的方式去吃,蒸饺子,煮饺子,煎饺子,想要用策略模式来设计这个场景,怎么弄?1.复习简单工厂模式具体的代码
- 学习过java基础,最近趁着大量课余时间想学习Android开发。百度很多资料Android studio,由Google开发的开发工具,那
- 相关阅读C#使用LINQ查询操作符实例代码(一)C#使用LINQ查询操作符实例代码(二)示例业务背景介绍示例参考《C#高级编程(第六版)》L
- 前言日志处理是每个项目当中一个非常重要的内容。没有了日志,也就失去了对系统的可控性。没有日志,系统出现任何问题,都会没有踪迹可寻,这对一个信
- 前言本篇内容:提示语的国际化返回,自定义多语言。本文使用aop方式,拦截接口返回的数据,进行转换。正文 先看这次示例教
- 时间处理相关类:1.java.util.Date:时间类2.java.text.DateFormat:时间格式化类(抽象类),实现类:jav
- 通过路径从磁盘直接读取图片这段时间在做Springboot和Vue的例子,读取图片给出路径直接可以读,太方便了,一直么有搞懂为什么。后面看到
- 一、概述我们对于这个图片肯定会非常熟悉,这两幅图片我们都可以看做是一个文件结构,对于这样的结构我们称之为树形结构。在数据结构中我们了解到可以
- 一、材料准备1、jdk1.8的安装包2、maven安装包3、idea工具二、配置jdk1.8环境变量1.jdk下载jdk下载网址(点击此链接
- poi解析Excel文件版本问题解决办法poi解析Excel文件时有两种格式: HSSFWorkbook格式用来解析Excel2003(xl
- 就我们所知道的,java中有子类和父类,子类由于继承父类而形成,那么父类还有没有父类呢?答案是有了,父类的父类就是object类,一切父类都
- 下面是一个邮件接收的工具类,有点长!!!public class ReciveMail { private MimeMessage msg
- 打jar包实现分离依赖lib和配置为了业务需要配置文件和jar分离,便于使用者修改配置信息,在网上找了很久,最终找到一个简单有效的方法。操作
- 最近在学习AngularJS的知识,收获不少,不过因为自己平时工作时开发都是用的freemarker+springmvc来做的页面数据交互,
- 方法一(推荐):设置datetimepicker的属性ShowCheckBox为true在窗口初始化时候,添加代码this.datetime
- 1|0构造socket在【客户端/服务端】的通信模式中,客户端需要主动构造与服务器连接的 Socket,构造方法有以下几种重载形式:Sock
- Unity IPostBuildPlayerScriptDLLsUnity IPostBuildPlayerScriptDLLs是Unity
- 开篇:我们将前面的springboot整合H2内存数据库,实现单元测试与数据库无关性提供的Restful服务注册到spring cloud的