Andrid异步更新UI:Handler(二)深入了解:Message你真的会创建?它是如何子线程和主线程通知?

news2025/1/11 0:03:12

目录

  1. 为什么会有Handler
  2. Handler的原理,以及对象讲解
  3. 主线程的loop在哪里,为什么主线程loop没有阻塞呢?
  4. Looper如何保证唯一
  5. Handler为什么会引发内存泄漏呢?
  6. Message应该如何创建它?


一、为什么会有Handler


线程分为主线程(主线程又叫UI线程)和子线程,主线程即ActivityThread,规定只有此线程能操作UI,但我们从后台请求数据,都是在子线程操作,所以需要有人帮忙把线程切换一下,所以就有了Handler。


二、Handler的原理,以及对象讲解


2.1 它是如何子线程和主线程通知?

我们来看看Handler的最最最初的实现。主线程和子线程,子线程的数据需要通知到主线程去,主线程拿到数据以后,就去调用方法去更新UI。那么我们想想,如果你去实现子线程通知主线程,也就是两个线程之间通讯,那么你会使用什么方法?

在这里插入图片描述

使用全局变量!共享成员,比如:

//这里只是简单的介绍了一下
var flag:Boolean = false
Thread(object :Runnable{
            override fun run() {
                if (flag) {
                    //调用通知UI更新的方法
                }
            }
        }).start()
        
        Thread(object :Runnable{
            override fun run() {
                flag = true
            }
        }).start()

如下,sendMessage就是上面的一个变化通知,当收到新消息的时候,则会回调handleMessage方法进行操作。

	
handler.sendMessage(uimsg);

final Handler handler = new Handler() {
        @Override
        public void handleMessage(Message msg) {
            getMessage(msg);
        }
};

2.2 对象讲解:looper,message,messagequeue

Handler当然没有我们上面说的那么简单,只是我们拆开了最初的原理。接下来我们再讲一下Looper对象,我们可以注意到,我们写的最初的线程,执行完就结束了,那么肯定不能结束呀,对吧,因为他要继续监听消息。所以呢,还应该进行如下这样的改进,让它不断的循环:

var flag:Boolean = false
        Thread(object :Runnable{
            override fun run() {
                while (run){
                    if (flag) {
                        //调用通知UI更新的方法
                    }
                }
                
            }
        }).start()

        Thread(object :Runnable{
            override fun run() {
                flag = true
               
            }
        }).start()

而对应到Handler,Looper就是做这个事情的,我们可以看看源码就会有一个循环,不断的循环处理消息:
在这里插入图片描述因为消息很多(message),所以我们会有一个MessageQueue通道来进行管理、传输以及通知。

消息来了以后,loop收到以后开始发送,loop有一个for循环的功能。这里注意一个问题,loop是已经开始运转了,还是收到消息才开始启动呢?

在这里插入图片描述

没错,是已经运转了。线程一开启,就立马启动了。上面的例子我们可以看到,for已经循环了。

我们可以看看调用handleMessage的源码:
在这里插入图片描述looper在循环,如果有数据则开始调用回调,而通知的方法则在handler里面。

在这里插入图片描述源码的主要方法调用:
在这里插入图片描述
流程示意图:
在这里插入图片描述

2.3 这里有一个疑问,loop是如何知道线程是谁的,queue又是谁的

直接在handler里面创建了looper,所以就知道是谁的了。
在这里插入图片描述而queue则是在looper里面创建的。

在这里插入图片描述


三、主线程的loop在哪里,为什么主线程loop没有阻塞呢?


主线程的loop在哪里,在main方法,ActivityThread里面:

在这里插入图片描述

在这里插入图片描述我们可以注意到,主线程也不会结束,mian方法也有一个loop循环。

在这里插入图片描述
为什么主线程loop没有阻塞呢?但我们看到,其实他还是阻塞的,不阻塞,程序就运行完成了。只不过,他是如何阻塞的?他是有数据刷新才运行,没有数据刷新就阻塞,让出cpu,所以他采用的队列、epoll阻塞,去执行其他的任务,比如刷新ui。



四、Looper如何保证唯一


一个线程只能有一个loop,不可能有多个,因为你调用了looper的loop,后面的代码就不会执行了。
调用两次prepare会崩溃,为什么会崩溃呢?因为一个线程只能有一个looper,源码里面直接报错了:
在这里插入图片描述在这里插入图片描述判断有没有值。有,直接报错,没有就往下执行。这就是它的执行方式。
在这里插入图片描述


五、Handler为什么会引发内存泄漏呢?


Handler是内部类,内部类持有外部类的引用。但为什么其他内部类就可以呢?持有链。长生命周期,持有短生命周期的引用,就会导致无法被释放。

5.1 持有链

首先我们先想想,哪个对象不可能被释放?

在这里插入图片描述

没错,主线程对象。然后主线程对象持有looper,而looper持有messagequeue,而messagequeue持有message,message是一个链表,而message又持有hander,而handler又持有activity,所以长生命周期持有短生命周期activity,那么页面切换,activity就无法得到释放。

在这里插入图片描述在这里插入图片描述

六、Message应该如何创建它?


在这里插入图片描述为什么用2呢?这里要了解一下内存抖动。用第一种,会导致创建大量的对象并销毁对象就是内存抖动。如下这种就是内存抖动。
在这里插入图片描述
解决内存抖动的根本方法,就是内存复用,对象复用。所以我们要调用obtain方法。message是一个链表,存储着message进行复用。
在这里插入图片描述
这里的spool就是一个message。如果你直接new message,那么每次都是一个新的链表,而使用Message的obtain方法,它会进行复用。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2065213.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

软件工程造价师习题练习 18

1.在软件估算过程中,我们主要对项目的规模,质量,进度和成本进行估算 错误 正确 软件估算不对质量进行估算,只对项目的规模,进度,成本进行估算。 答案:错误 2.在使用类比法进行估算时&#xf…

基于单片机的电子指南针设计

摘要:随着科技的进步与发展,尤其是电子技术的推动下社会逐渐向智能化、信息化和人工智能等方面发展。与此同时市场更加需要掌握精确的方位角度信息,服务到生活、产业、出行、航行等各方各面。传统的指南针由于本身的物理特性,抗干…

深度剖析C++string(上篇)

目录 前言 1.C string类 2.string类中的常见构造 3.string类对象的容量操作 4.. string类对象的访问及遍历操作 5. auto和范围for(补充) auto关键字 范围for 结束语 前言 C语言我们学习了字符串和字符串的相关函数,在C语言中,字符串是…

【Redis基础1】——使用场景-缓存穿透

目录 1-引言:初始缓存穿透1-1 缓存穿透是什么?(What)1-2 缓存穿透是怎么造成的?(Why) 2- 核心:如何避免缓存穿透(How)2-1 方案1:缓存空数据2-2 方案2:布隆过滤器2-2-1 布隆过滤器原理…

计算机毕业设计Hive+Hadoop深圳共享单车预测系统 共享单车数据分析可视化大屏 共享单车爬虫 共享单车数据仓库 机器学习 深度学习 PySpark

步骤: 1.Python采集深圳政府公开数据平台的共享单车数据(最大可采集上亿2017-2021数据),并用百度逆地理编码服务解析经纬度获取位置信息。并将数据上传hdfs; 2.可使用sklearn、卷积神经网络等算法对数据进行分析,对共享单车的订单…

树链剖分学习

1.算法适用:维护树上一段或者某个点的子树权值和,同时支持树上路径大小修改,子树修改 2.实现:将树剖分成一条一条链,重链和轻链。其中,轻链连接非重儿子,重链相反。 3.主体部分: …

Array List 练习(添加手机对象并返回要求的数据)

package ArrayListDemo;import java.util.ArrayList;public class ArrayListDemo7 {public static void main(String[] args) {//1.创建集合对象ArrayList<Phone> list new ArrayList<Phone>();//2.创建手机对象Phone ph1 new Phone("小米",1000);Pho…

06-07 - 文件系统(下)

---- 整理自 王利涛老师 课程 实验环境&#xff1a;宅学部落 www.zhaixue.cc 文章目录 15. 根文件系统15.1 根文件系统的基本概念15.2 挂载根文件系统的三种方式15.3 根文件系统的挂载15.3.1 rootfs 文件系统的概念15.3.2 root 参数分析15.3.3 根目录“ / ” 16. 使用 initrd 作…

智能家居:‌未来生活的崭新篇章

随着科技的飞速发展&#xff0c;‌智能家居正逐渐从概念走向现实&#xff0c;‌为我们的日常生活带来前所未有的便捷与舒适。‌智能家居&#xff0c;‌简而言之&#xff0c;‌就是利用先进的物联网技术、‌人工智能、‌大数据分析等&#xff0c;‌将家中的各种设备连接起来&…

text token通过cross attention来编辑图片

https://zhuanlan.zhihu.com/p/645249016https://zhuanlan.zhihu.com/p/645249016diffusers中的cross-attention实现_diffusers.models.attention.crossattention-CSDN博客文章浏览阅读71次。在models/unets/unet_2d_condition.py中,如何在Unet中将原注意力类替换,Unet中的at…

笔记:Echarts柱状图 实现滚轮条 数据太多

效果&#x1f447;&#x1f447;&#x1f447; 配置&#xff1a;&#x1f447; let option {dataZoom: [{type: "slider",show: true,zoomLock: true,start: 0,end: 20,bottom: 60,height: 10,textStyle: {color: "transparent",fontSize: 9,},fillerColo…

WEB应用服务器 -- Tomcat详解及案例实现

一、Web前端三大核心技术 1.1 HTML HTML&#xff08;HyperText Markup Language&#xff09;超文本标记语言&#xff0c;它不同于一般的编程语言。超文本即超出纯文本的范畴&#xff0c;例如&#xff1a;描述文本颜色、大小、字体等信息&#xff0c;或使用图片、音频、视频等…

Linux 操作系统全览:从文件管理到系统状态掌控

在当今的计算机技术领域&#xff0c;Linux 操作系统以其稳定性、安全性和高度的可定制性而备受青睐。本文将深入探讨 Linux 操作系统的多个重要方面&#xff0c;包括文件管理、文本操作、软件安装、用户和权限管理以及网络和系统状态管理。 一、Linux 文件管理 Linux 的文件系…

哈工深、NUS等联合提出全新信息抽取基准任务:细粒度定位的统一多模态信息抽取...

信息抽取是 NLP 领域长久以来最经典的研究方向之一。信息抽取的研究经历过了多模态抽取阶段&#xff0c;以及大一统的抽取阶段。在大语言模型和多模态时代&#xff0c;信息抽取的发展何去何从&#xff0c;目前社区仍在积极探索中。 近期&#xff0c;由哈工深和新加坡国立大学等…

131-横向移动-Kerberos攻击SPN扫描WinRMWinRSRDP

1、RDP协议 Remote Desktop Protocol 远程桌面协议通常开放3389 &#xff0c;Windows上面使用mstsc就可以弹出最常见的远程桌面连接方式&#xff0c;一般都是使用明文进行连接其实还可以使用hash进行 在内网中使用RDP协议一般是需要进行代理转发或者建立节点的 端口扫描 shel…

实现AOP机制 + Spring总结

文章目录 1.目录2.SmartAnimal.java 接口&#xff08;JDK代理必须有接口&#xff09;3.SmartDog.java4.SmartAnimalAspect.java5.SunSpringApplicationContext.java1.在Bean的后置处理器之后使用动态代理2.完整代码 6.测试1.AppMain.java 调用被代理的类的方法2.结果 7.Spring底…

维特——六九轴陀螺仪传感器代码的移植方法

使用的是如图所示的这款陀螺仪&#xff0c;相比MPU6050&#xff0c;它可以做到Z轴不漂的效果。 官方给的代码是使用两个串口&#xff0c;一个用来和上位机通信&#xff0c;一个用来于该模块通信。在实际使用中&#xff0c;我们只需要单片机和该模块通信即可&#xff0c;所以我…

Centos安装Jenkins教程详解版(JDK8+Jenkins2.346.1)

本教程基于 JDK8 和 Jenkins2.346.1 JDK安装 下载OpenJDK8文件 wget https://mirrors.tuna.tsinghua.edu.cn/Adoptium/8/jdk/x64/linux/OpenJDK8U-jdk_x64_linux_hotspot_8u422b05.tar.gz解压到指定目录 # 创建目录 mkdir -p /usr/local/software# 解压文件到指定目录&#…

uniapp点击预览图片,两种效果

背景&#xff1a; 在使用uniapp框架中&#xff0c;我们对图片的展示需要点击放大展示(单张)&#xff1b;如果是多张图片&#xff0c;要支持左右滑动查看多张图片(多张)。 官网链接&#xff1a;点击跳转官网 一、单张&#xff0c;点击放大 代码&#xff1a; <template> …

Spring模块详解Ⅲ(Spring Expression Language (SPEL)和Spring JDBC)

目录 Spring Expression Language&#xff08;SpEL&#xff09;Spring Context 模块详解1. 什么是 Spring Context?2. ApplicationContext 的主要实现3. 国际化支持国际化消息文件使用 MessageSourceXML 配置&#xff1a;Java 配置&#xff1a; 使用国际化消息&#xff1a; 4.…