[Java、Android面试]_18_详解Handler机制 常见handler面试题(非常重要,非常高频!!)

news/2024/5/21 15:54:55

本人今年参加了很多面试,也有幸拿到了一些大厂的offer,整理了众多面试资料,后续还会分享众多面试资料。
整理成了面试系列,由于时间有限,每天整理一点,后续会陆续分享出来,感兴趣的朋友可关注+收藏

文章目录

    • 1. MessageQueue、Looper和Handler之间的关系:
    • 2. Handler运行机制:
    • 3. Handler机制详解
      • 3.1 子线程到主线程通信方式有哪些?子线程到主线程通信的原理?
      • 3.2 Handler内存泄露的原因?
      • 3.3 MessageQueue中存储的Msg数量有上限吗?为什么?能不能用阻塞队列做MessageQueue?
      • 3.4 Handler如何处理发送延迟消息的?
      • 3.5 使用Message时应该如何创建它?
      • 3.6 Handler没有消息处理时,会阻塞吗?阻塞后为什么不会产生ANR?
      • 3.7 如何在子线程中创建handler?

Handler机制主要包括:MessageQueue、Looper、Handler以及Message四个部分。
· Message: 传递的消息及数据
· MessageQueue: 消息队列,但是它的内部并不是用的队列实现的,而是通过单链表实现的,因为单链表在插入和删除上更有优势,主要功能是向消息池投递消息(MessageQueue.enqueueMessage)和从消息池取走消息(MessageQueue.next).
· Handler: 消息辅助类,主要功能是向消息池发送各种消息(Handler.sendMessage)和处理相应的消息事件(handler.handleMessage())。
· Looper:消息控制器,不断循环执行(Looper.loop),从MessageQueue中读取消息,按分发机制将消息分发给目标处理者。
在这里插入图片描述

1. MessageQueue、Looper和Handler之间的关系:

(1)每个线程只有一个Looper,Looper是保存在ThreadLocal中的。主线程已经创建了一个Looper了,所以不需要创建Looper,如果是其他线程,就需要创建Looper;
(2)每个线程中可以有多个Handler,即一个Looper可以处理来自多个Handler的消息。怎样区分message来自哪个Handler?message上面有个属性是target,这个target就会标识Handler.
(3)Looper中维护一个MessageQueue,MessageQueue中的message来自不同的Handler.
在这里插入图片描述

2. Handler运行机制:

在子线程执行完耗时操作,当Handler发送消息时,将会调用 MessageQueue.enqueueMessage,向消息队列中添加消息。 当通过 Looper.loop开启循环后,会不断地从消息池中读取消息,即调用 MessageQueue.next, 然后调用目标Handler(即发送该消息的Handler)的 dispatchMessage方法传递消息, 然后返回到Handler所在线程,目标Handler收到消息,调用 handleMessage方法,接收消息,处理消息。

3. Handler机制详解

3.1 子线程到主线程通信方式有哪些?子线程到主线程通信的原理?

子线程到主线程的通信方式有:Rxjava、eventBus和Handler,但底层原理都是通过Handler来实现的。
通信原理:
子线程将消息送入队列:handler.sendMessage(msg) -> messagesQueue.enqueueMessage()

主线程:main()-> looper.loop() -> MessageQueue.next() -> handler.dispatchMessage-> handler.handleMessage()。

主线程中开启后,就会一直执行looper.loop(),不断的同MessageQueue中获取消息并通过msg上的target标志分发到对应的Handler去。

其本质是是通过内存共享的方式,MessageQueue就像是那个共享的内存。

3.2 Handler内存泄露的原因?

可参考:http://t.csdn.cn/gVsNw
当使用内部类来创建Handler时,handler对象会隐式地持有一个外部类对象(通常是一个Activity)的引用(这是JAVA的特性)。而Handler通常会伴随着一个耗时的后台线程(例如从网络拉取图片)一起出现,这个后台线程在任务执行完毕(例如图片下载完毕)之后,通过消息机制通知Handler,然后Handler把图片更新到界面。然而,如果用户在网络请求过程中关闭了Activity,正常情况下,Activity不再被使用,它就有可能在GC检查时被回收掉,但由于这时线程尚未执行完,而该线程持有Handler的引用(不然它怎么发消息给Handler?),这个Handler又持有Activity的引用,就导致该Activity无法被回收(即内存泄露),直到网络请求结束。

1)那为什么不会被回收呢?可以分析分析引用链
Handler -> Activity: handler持有activity的引用
那么谁持有hander的引用呢?源码中有msg.target=handler, 所以:msg->handler;
而messageQueue持有msg的引用,即:messageQueue -> msg;
Lopper持有MessageQueue,Looper -> messageQueue
那么谁持有Looper的引用呢?从源码得知,sThreadLocal->Looper,
而sThreadLocal是new的一个静态变量,可以进行作为GC root,所以整个引用链:
sThreadLocal->Looper->messageQueue->msg->hander->activity.
而sThreadLocal是GCroot,所以整个链都在GCroot上,不会被回收,所以出现上面那种情况时,就出现了内存泄露。

2) 怎样解决Handler导致的内存泄露?
方法1:静态内部类+弱引用

1.static class MyHandler extends Handler {
2.    WeakReference<Activity > mActivityReference;
3.    MyHandler(Activity activity) {
4.        mActivityReference= new WeakReference<Activity>(activity);
5.    }
6.    @Override
7.    public void handleMessage(Message msg) {
8.        final Activity activity = mActivityReference.get();
9.        if (activity != null) {
10.            //...
11.        }
12.    }
13.}

将 Handler 声明为静态内部类,Handler 也就不再持有 Activity 的引用,所以 Activity 可以随便被回收。但Handler 不再持有 Activity 的引用,导致 Handler 无法操作 Activity 中对象,所以可以在 Handler 中添加一个对 Activity 的弱引用(WeakReference)。

方法2:程序逻辑保护
在 Activity 被销毁时及时清除消息,从而及时回收 msg和handler,使用removeMessage()和判断mHandler来继续移除msg和handler,避免内存泄漏问题。如:

1.@Override
2.protected void onDestroy() {
3.    super.onDestroy();
4.    if (mHandler != null)  {
5.        mHandler.removeCallbacksAndMessages(null);
6.    }
7.}

3.3 MessageQueue中存储的Msg数量有上限吗?为什么?能不能用阻塞队列做MessageQueue?

MessageQueue没有上限。原因如下:
· 从代码的角度,主线程中的looper.loop()是写的一个死循环,如果有上限,在入队和出队时肯定会做额外判断,然而并没有,所以是没有上限。

· 从应用分析的角度:如下图,从handler源码可以看出,AMS对activity、Service等和管理,以及activity对fragment的管理都是转换为msg,然后使用handler完成的。我们无法判断出用户会开启多少fragemnt、activity,所以是不会设置上限的。
在这里插入图片描述
不能使用阻塞队列,阻塞队列是在队列达到上限或下限后进行阻塞,而这没有上限,所以不能使用阻塞队列。

3.4 Handler如何处理发送延迟消息的?

消息入队阶段:消息队列是一个单向链表,会根据执行时间进行排序;由源码可知,在使用sendMessageDelayed(msg, delayMillis)时,会添加一个延迟时间,其源码如下:
在这里插入图片描述
而MessageQueue会根据时间进行排序,如下所示:
在这里插入图片描述
在取出消息的执行阶段:每次都会从链表表头去消息,每次取出消息的会判断当前时间是否小于Msg的执行时间,如果小于,则调用一个native函数nextPollTimeoutMills()进行延迟等待。如果大于,则说明到了执行时间,则进行之后的操作。其源码如下:
在这里插入图片描述

3.5 使用Message时应该如何创建它?

尽量不要通过new Message的方式, 可以通过如下方式创建:

1.Message message = myHandler.obtainMessage();  //通过 Handler 实例获取,底层也是调用的Message.obtain()
2.Message message1 = Message.obtain();          //通过 Message 获取

原因如下:
系统维护了一个消息池,每次用完的消息并不会直接将消息销毁,而是将内存全部置空,然后放到消息池去。

这样的好处就是为了避免”内存抖动“从而导致OOM和卡顿,因为内存抖动就是频繁的申请内存和释放内存。系统会处理大量的消息,如果都使用new的方式,就会造成内存抖动。

3.6 Handler没有消息处理时,会阻塞吗?阻塞后为什么不会产生ANR?

会发生阻塞,但不会产生ANR。
因为ANR是由于超时了,才会ANR。其超时阈值如下:
在这里插入图片描述

3.7 如何在子线程中创建handler?

在这里插入图片描述
在这里插入图片描述
在子线程中创建handler,需要创建Looper(主线程中不用创建是因为启动时就创建好了),所以可以继承Thread类,里面定义好Looper的创建。

没有消息时, looper也会一直轮询。


http://www.mrgr.cn/p/57474351

相关文章

结对编程-c++四则运算

题目: 小学老师要每周给同学出300道四则运算练习题。 –这个程序有很多种实现方式: C/C++ C#/VB.net/Java Excel Unix Shell Emacs/Powershell/Vbscript Perl Python –两个运算符,100 以内的数字,不需要写答案。 –需要检查答案是否正确,并且保证答案在 0..100 之间 –尽可…

重载全局的new和delete

重载全局的new和delete ::operator new ::operator new[] -> 不可以被声明与同一个namespace之内 new会执行三个动作: -> 之前的代码提到:new本身会开辟内存空间.所以声明方法需要一个size_t size的参数inline void* operator new(size_t size) {}::operator delete ::op…

django celery 异步任务 异步存储

环境&#xff1a;win11、python 3.9.2、django 4.2.11、celery 4.4.7、MySQL 8.1、redis 3.0 背景&#xff1a;基于django框架的大量任务实现&#xff0c;并且需要保存数据库 时间&#xff1a;20240409 说明&#xff1a;异步爬取小说&#xff0c;并将其保存到数据库 1、创建…

QT 串口助手 学习制作记录

QT 串口助手qt 学习制作记录 参考教程&#xff1a;​​​​​​QT初体验&#xff1a;手把手带你写一个自己的串口助手_qt设计串口助手的流程图-CSDN博客 Qt之串口编程&#xff08;添加QSerialPort模块&#xff09;_如何安装 qt串口模块教程-CSDN博客 串口调试助手&#xff1…

Markdown的基本语法

Markdown的基本语法 参考来源:Markdown 教程 、手把手教会你使用Markdown 、Cmd Markdown 01、标题 1.1、# 号使用 使用 # 号可以表示 1-6 级标题,一级标题对应一个 # 号,随着 # 的个数递增,一级标题字号最大,六级标题字号最小。 代码如下: # 一级标题 ## 二级标题 ### 三…

Swoole 实践篇之结合 WebRTC 实现音视频实时通信方案

原文首发链接:Swoole 实践篇之结合 WebRTC 实现音视频实时通信方案 大家好,我是码农先森。 引言 这次实现音视频实时通信的方案是基于 WebRTC 技术的,它是一种点对点的通信技术,通过浏览器之间建立对等连接,实现音频和视频流数据的传输。 在 WebRTC 技术中通常使用 WebSoc…

【数据结构与算法】:10道链表经典OJ

目录 1. 移除链表元素2. 反转链表2.1反转指针法2.2 头插法 3. 合并两个有序链表4. 分隔链表5. 环形链表6. 链表的中间节点7. 链表中倒数第K个节点8. 相交链表9. 环形链表的约瑟夫问题10. 链表的回文结构 1. 移除链表元素 思路1&#xff1a;遍历原链表&#xff0c;将 val 所在的…

JS:locaton.href=易错点

这个地方我错了两次了,每次还让人找得焦头烂额,或许记下来能让我长点记性 易错写法: 正确写法: 不知道为什么我每次都写成location.href("#"),明明应该是location.href="#",而且因为它并不报错,我这笨脑瓜差错真的很心累

《战神4》和《战神5》有什么联系吗 苹果电脑如何运行《战神4》苹果电脑玩战神 Mac玩游戏 战神5攻略 crossover激活码

《战神4》&#xff08;God of War 2018&#xff09;和《战神5》&#xff08;God of War: Ragnark&#xff09;是一对引人注目的游戏作品&#xff0c;它们不仅在游戏界引起了广泛的关注&#xff0c;也给玩家带来了深入探索北欧神话世界的机会。这两部游戏之间的联系不仅体现在剧…

Codeforces Round 937 (Div. 4) VP记录

距离退役又近了一步!第一次 VP 比赛(也是第一次打 CF)。 感到自己距离退役又近了一步。 A. Stair, Peak, or Neither? 题意 You are given three digits \(a\), \(b\), and \(c\). Determine whether they form a stair, a peak, or neither.A stair satisfies the conditi…

前端使用 Konva 实现可视化设计器(5)

关于第三章提到的 selectingNodesArea,在后续的实现中已经精简掉了。 而 transformer 的 dragBoundFunc 中的逻辑,也直接移动 transformer 的 dragmove 事件中处理。请大家动动小手,给我一个免费的 Star 吧~ 这一章花了比较多的时间调试,创作不易~ github源码 gitee源码 示…

蓝桥杯 每天2题 day6

碎碎念&#xff1a;哇咔咔 要不是中间缺勤一天就圆满day7了&#xff01;最后一晚上&#xff01;写题复习哇咔咔 唉&#xff0c;睡了一觉就看不下去了&#xff0c;&#xff0c;&#xff0c;看看之前的笔记洗洗睡觉&#xff0c;&#xff0c;&#xff0c; 记得打印准考证带好东西…

C语言:约瑟夫环问题详解

前言 哈喽&#xff0c;宝子们&#xff01;本期为大家带来一道C语言循环链表的经典算法题&#xff08;约瑟夫环&#xff09;。 目录 1.什么是约瑟夫环2.解决方案思路3.创建链表头结点4.创建循环链表5.删除链表6.完整代码实现 1.什么是约瑟夫环 据说著名历史学家Josephus有过以下…

CSS aspect-ratio属性设置元素宽高比

aspect-ratio 是CSS的一个属性&#xff0c;用于设置元素的期望宽高比。它设置确保元素保持特定的比例&#xff0c;不受其内容或容器大小的影响。 语法&#xff1a; aspect-ratio: <ratio>;其中 <ratio> 是一个由斜杠&#xff08;/&#xff09;分隔的两个数字&…

在一台恢复测试机器上验证oracle备份有效性

一 目的 定期将生产环境oracle数据库恢复到一台测试环境数据库服务器上&#xff0c;以验证备份是否有效&#xff0c;是否能正常恢复。 二 环境 这里以恢复orcl1库为例&#xff0c;计划在orcl这个实例上进行恢复测试。 三 实验步骤 3.1 在目标端创建和源端一样的备份目录 ①…

玩家——玩家格挡

目的:玩家死亡动画调用 创建护盾蓝图 玩家角色蓝图中编写护盾启用逻辑 创建护盾CDUI 玩家角色蓝图中护盾CD1.玩家死亡动画的调用2.创建护盾蓝图3.玩家角色蓝图中编写护盾启用逻辑4.创建护盾CDUI 5.玩家角色蓝图中实现护盾CD 初始化UI本文来自博客园,作者:荒坂株式会社,博客…

2024年nodejs调用小红书最新关注(粉丝)follow接口,api接口分析2004-04-16

一、打开chrome按f12&#xff0c;点击右上角的“关注”按钮&#xff0c;抓包位置如下&#xff1a; (图1 follow接口) 二、follow接口分析 1、请求地址 https://edith.xiaohongshu.com/api/sns/web/v1/user/follow 2、请求方法: POST 3、请求头&#xff1a; :authority: edith…

吴恩达机器学习-第二课-第二周

吴恩达机器学习 学习视频参考b站:吴恩达机器学习 本文是参照视频学习的随手笔记,便于后续回顾。 TensorFlow实现神经网络模型训练细节 训练步骤与和逻辑回归的比较 训练模型分为三步: 1.确定f(x)函数 2.确定损失函数和代价函数 3.通过数据训练寻找最小值详细介绍 1.创建模型…

分析ARP解析过程

一、实验环境 主机A和主机B连接到交换机&#xff0c;并与一台路由器互连&#xff0c;如图7.17所示&#xff0c;路由器充当网关。 图7.17 二、需求描述 查看 ARP 相关信息,熟悉在PC 和 Cisco 设备上的常用命令,设置主机A和主机B为同一个网段网关设置为路由接口地址。 三、推…

全局视角观看Python备忘录-英文版

全局视角观看Python备忘录-英文版