38. 【Android教程】Handler 消息传递机制

跑在主线程(即UI线程)当中的,而且所有的 UI 刷新以及输入处理必须在主线程中执行。这样一旦任务多了就会阻塞 UI 线程导致画面卡顿,从而严重影响性能,所以正确的做法是将耗时的操作单独放在子线程中与 UI 线程隔离,等到耗时操作完成之后再把结果传到 UI 线程进行展示,这就要用到本节学到的消息传递工具——Handler。

1. Handler 基本原理

Handler 是连接不同线程的管道,它让你能够在不同线程之间自由的传递数据,当然我们用的比较多的场景是在子线程中 与主线程通信。因为 Android 系统要求只能在主线程操作 UI,所以常规的做法是将子线程耗时操作的结果传递到 UI 线程进行刷新。

Handler 的基本原理如下图所示:

每一个 Handler 实例与一个线程关联,每一个线程又会维护一个自己的 MessageQueue,当我们创建一个 Handler 的时候我们需要制定一个 Looper 对象(Looper对象对应一个线程),这样就将一个 Handler 对象和一个线程绑定到了一起,随后就可以编写我们的耗时操作,然后通过 Handler 将消息塞入线程的 MessageQueue中,当对应线程从 MessageQueue 取出该条消息的时候,就会回调 Handler 的 handleMessage方法并拿到消息,这样就完成了跨线程通信。

2. Handler 相关方法介绍

  • void handleMessage(Message msg):
    在该方法中处理其他线程传递过来的消息*(用的非常多,一定要掌握!)*
  • sendEmptyMessage(int what):
    发送一条空消息,what 可以理解为消息 ID
  • sendEmptyMessageDelayed(int what,long delayMillis):
    延时发送空消息,what 为自定义 ID
  • sendMessage(Message msg):
    发送消息,msg 是消息内容
  • sendMessageDelayed(Message msg):
    延时发送,msg是消息内容
  • hasMessage(int what):
    检查 MessageQueue 中是否包含一条 ID 为 what 的消息。what 是用户自定义的整形数,可作为消息 ID

3. Handler 使用示例

讲了这么多理论知识,我们来通过一个示例来演示一下如何通过 Handler 完成线程通信。一个很常见的场景就是当 App 需要执行一个耗时任务的时候,会把任务放在子线程中执行,但是在主线程通过一个进度条(ProgressBar)来实时更新进度,这样让用户能够随时看到任务执行的进展。

3.1 布局文件

布局比较简单,主要由三个部分:

  • 启动任务
  • 进度文本
  • 进度条
<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"android:layout_width="match_parent"android:layout_height="match_parent"><ProgressBarandroid:id="@+id/progressBar"style="?android:attr/progressBarStyleHorizontal"android:layout_width="match_parent"android:layout_height="wrap_content"android:layout_alignParentStart="true"android:layout_alignParentTop="true" /><Buttonandroid:id="@+id/start_progress"android:layout_width="wrap_content"android:layout_height="wrap_content"android:layout_below="@+id/progressBar"android:layout_alignParentStart="true"android:layout_marginStart="24dp"android:layout_marginTop="62dp"android:text="开始任务" /><TextViewandroid:id="@+id/textView"android:layout_width="wrap_content"android:layout_height="wrap_content"android:layout_alignTop="@+id/start_progress"android:layout_alignBottom="@+id/start_progress"android:layout_alignParentEnd="true"android:layout_marginEnd="85dp"android:gravity="center"android:text="当前进度:0%"android:textSize="16sp" /></RelativeLayout>

3.2 MainActivity

主要的逻辑全在 MainActivity 中实现,要完成以下 3 个任务:

  • 线程切换: 在子线程中执行耗时操作
  • 更新进度: 在主线程更新ProgressBar,并同步更新TextView
  • 消息传递: 通过Handler将进度从子线程传递到主线程

基本完成了上面 3 个任务,整个线程通信就搞定了。代码如下:

package com.emercy.myapplication;import android.app.Activity;
import android.os.Bundle;
import android.os.Handler;
import android.os.Message;
import android.view.View;
import android.widget.Button;
import android.widget.ProgressBar;
import android.widget.TextView;public class MainActivity extends Activity {private static final int MAX = 100;private static final int START_PROGRESS = 100;private static final int UPDATE_COUNT = 200;private ProgressBar progressBar;private Button startProgress;private TextView textView;private boolean mHasStart;// 任务2:在主线程刷新进度条Handler mHandlerThread = new Handler() {@Overridepublic void handleMessage(Message msg) {super.handleMessage(msg);if (msg.what == START_PROGRESS) {if (!mHasStart) {thread.start();mHasStart = true;}} else if (msg.what == UPDATE_COUNT) {textView.setText("当前进度:" + msg.arg1 + "%");progressBar.setProgress(msg.arg1);}}};// 任务1:在子线程执行耗时操作,通过sleep模拟耗时任务Thread thread = new Thread(new Runnable() {@Overridepublic void run() {for (int i = 0; i <= 100; i++) {try {// 一秒钟的耗时操作Thread.sleep(1000);} catch (InterruptedException e) {e.printStackTrace();}Message message = new Message();message.what = UPDATE_COUNT;message.arg1 = i;mHandlerThread.sendMessage(message);}}});@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);progressBar = findViewById(R.id.progressBar);startProgress = findViewById(R.id.start_progress);textView = findViewById(R.id.textView);progressBar.setMax(MAX);startProgress.setOnClickListener(new View.OnClickListener() {@Overridepublic void onClick(View view) {// 任务3:通过Handler传递进度消息Message message = new Message();message.what = START_PROGRESS;mHandlerThread.sendEmptyMessage(START_PROGRESS);}});}
}

4. 小结

本节学习了 Android 消息传递机制,详细介绍了 Handler 的基本原理,以及 Looper、线程、MessageQueue、Message 等概念。Handler 最常见的用途就是在子线程执行耗时操作的时候与主线程通信,通知主线程进行 UI 的刷新。一个完成的线程通信主要有 3 大任务,并用一个完成的示例演示了如何完成这 3 个任务,这一节的内容非常重要,请大家务必掌握!

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

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

相关文章

本地代码配置多个远程仓库进行推送

Git配置多个远程仓库 问题解决办法新增远程地址推送 问题 目前一个项目正在一个仓库中存储&#xff0c;需要新增一个仓库&#xff0c;实现能同时推送到两个仓库中&#xff0c;比如一个项目同时维护在github和gitee上。 解决办法 新增远程地址 直接在本地项目根目录下输入: …

玩原神玩的!30本提升你视野、眼界和格局的好书不如你挑的一本适合自己的书!——早读(逆天打工人爬取热门微信文章解读)

许久不见&#xff0c;雨天坐公车&#xff0c;别是一番滋味在心头 引言Python 代码第一篇 洞见 人民日报推荐&#xff1a;30本提升你视野、眼界和格局的好书第二篇 人民日报 来了&#xff01;新闻早班车要闻社会政策 结尾 不要着急 最好的总会在最不经意的时候出现 意外的六分钟…

Linux系统中Nginx的使用

Nginx是一款开源的高性能、高可靠性的Web服务器和反向代理服务器。它在Linux系统中得到了广泛的应用&#xff0c;被用于构建高性能的Web应用和提供反向代理服务。下面将介绍Nginx在Linux系统中的使用以及一些常见的应用案例。 一、Nginx的安装和配置 安装Nginx 在Linux系统中…

Excel如何计算时间差

HOUR(B1-A1)&"小时 "&MINUTE(B1-A1)&"分钟 "&SECOND(B1-A1)&"秒"

【C++】一篇文章带你深入了解list

目录 一、list的介绍二、 标准库中的list类2.1 list的常见接口说明2.1.1 list对象的常见构造2.1.1.1 [无参构造函数](https://legacy.cplusplus.com/reference/list/list/list/)2.1.1.2 [有参构造函数(构造并初始化n个val)](https://legacy.cplusplus.com/reference/list/list/…

TR5 - Transformer的位置编码

&#x1f368; 本文为&#x1f517;365天深度学习训练营 中的学习记录博客&#x1f356; 原作者&#xff1a;K同学啊 目录 前言什么是位置编码1. 定义2. 三角函数3. 位置编码公式4. 位置编码示例 可视化理解位置编码1. 代码实现2. 观察不同位置对应的曲线3. 整句话的位置编码可…

ffmpeg支持MP3编码的方法

目录 现象 解决办法 如果有编译包没有链接上的情况 现象 解决办法 在ffmpeg安装包目录下 &#xff0c;通过./configure --list-encoders 和 ./configure --list-decoders 命令可以看到&#xff0c;ffmpeg只支持mp3解码&#xff0c;但是不支持mp3编码。 上网查寻后发现&…

新的全息技术突破计算障碍

一种突破性的方法利用基于Lohmann透镜的衍射模型实时创建计算机生成全息图&#xff08;CGH&#xff09;&#xff0c;在保持3D可视化质量的同时&#xff0c;大大降低了计算负荷要求。 全息显示为制作逼真的三维图像提供了一条令人兴奋的途径&#xff0c;这种图像给人以连续深度…

【WEB前端2024】开源元宇宙:乔布斯3D纪念馆-第9课-摆件美化

【WEB前端2024】开源元宇宙&#xff1a;乔布斯3D纪念馆-第9课-摆件美化 使用dtns.network德塔世界&#xff08;开源的智体世界引擎&#xff09;&#xff0c;策划和设计《乔布斯超大型的开源3D纪念馆》的系列教程。dtns.network是一款主要由JavaScript编写的智体世界引擎&#…

合合信息Embedding模型:引领中文文本向量化技术新高度

目录 &#x1f345;前言&#x1f353;赛事含金量&#x1f353;Embedding技术简介&#x1f353;Embedding在大模型中的价值&#x1f353;合合信息Embedding模型特点及优势&#x1f353;合合信息Embedding模型测试&#x1f353;技术突破&#x1f353;公司介绍 &#x1f345;总结 …

Matplotlib官网查阅资料

Matplotlib官网详细的地址&#xff1a; 英文文档&#xff1a;https://matplotlib.org/stable/contents.html中文文档&#xff1a;https://www.matplotlib.org.cn/ Matplotlib英文官网: 查找属性&#xff1a; 1.进入官网。 2.查找参数属性。 Matplotlib中文官网: 查找属性:…

光纤、以太网电缆和 DSL 的比较:技术指南

了解光纤、以太网电缆和 DSL 之间的差异对于做出有关互联网连接的明智决策至关重要。本技术指南对这些技术进行了全面比较&#xff0c;讨论了它们的独特功能、性能指标和应用。它旨在为您提供必要的知识&#xff0c;以选择最适合您的特定需求的选项。 光纤、以太网电缆和 DSL …

Qwen1.5微调

引子 由于工作上需要&#xff0c;一直在用Qwen做大模型推理&#xff0c;有个再训练的需求&#xff0c;特此琢磨下Qwen的训练。OK&#xff0c;我们开始吧。 一、安装环境 查看显卡驱动版本 根据官网推荐 OK&#xff0c;docker在手&#xff0c;天下我有。 docker pull qwenll…

Recommended Azure Monitors

General This document describes the recommended Azure monitors which can be implemented in Azure cloud application subscriptions. SMT incident priority mapping The priority “Blocker” is mostly used by Developers to prioritize their tasks and its not a…

论文辅助笔记:处理geolife数据

论文笔记&#xff1a;Context-aware multi-head self-attentional neural network model fornext location prediction-CSDN博客 对应命令行里 python preprocessing/geolife.py 20 这一句 1 读取geolife数据 pfs, _ read_geolife(config["raw_geolife"], print_…

Spring的过滤器、拦截器、切面区别及案例分析

Spring的过滤器、拦截器、切面 三者的区别&#xff0c;以及对应案例分析 一、三者的实现方式 1.1 过滤器 xxxFilter 过滤器的配置比较简单&#xff0c;直接实现Filter接口即可&#xff0c;也可以通过WebFilter注解实现对特定URL的拦截&#xff0c;Filter接口中定义了三个方法…

工作记录:vue-grid-layout 修改 margin 导致 item 高度剧烈变化

问题 用 vue-gird-layout 时发现&#xff0c;当改变 margin 值时&#xff0c;item 的尺寸也会跟着变化。 如下图&#xff1a;row height 和每个 item 的 h 都保持不变。修改 margin-y&#xff0c;item 的实际高度也跟着变了&#xff1a; 原因 研究了一番&#xff0c;发现原…

如何查看全球历史影像

目录 示例 2024年3月28日 2022年9月21日 2021年11月3日 2020年11月18日 2019年5月15日 2017年2月27日 2016年12月20日 如何在ArcGIS中加载 如何查看全球历史影像&#xff0c;今天给大家分享一个可以在线直接访问查看全球历史影像的网站&#xff08;网址见文末&#xf…

flutter 设置启屏页 flutter_native_splash 坑记录

flutter_native_splash | Flutter packageCustomize Flutters default white native splash screen with background color and splash image. Supports dark mode, full screen, and more.https://pub.dev/packages/flutter_native_splash 发现一直白屏 原因是 代码中 下面…

前端路由的实现原理

当谈到前端路由时&#xff0c;指的是在前端应用中管理页面导航和URL的机制。前端路由使得单页应用&#xff08;Single-Page Application&#xff0c;SPA&#xff09;能够在用户与应用交互时动态地加载不同的视图&#xff0c;而无需每次都重新加载整个页面。 在前端开发中&…