Android HandlerThread 基础

HandlerThread

    • **一、HandlerThread的基本概念和用途**
      • 1. **目的**
      • 2. **与普通线程的区别**
    • **二、HandlerThread的使用步骤**
      • 1. **创建HandlerThread对象并启动线程**
      • 2. **创建Handler并关联到HandlerThread的消息队列**
      • 3. **发送消息到HandlerThread的消息队列**
    • **三、HandlerThread的生命周期和注意事项**
      • 1. **生命周期**
      • 2. **注意事项**
    • 四、使用 HandlerThread 和 线程池 举同一个例子
      • 1. **使用HandlerThread的示例:下载文件并更新UI**
      • 2. **使用线程池的示例:下载文件并更新UI(同样的功能)**
      • 3.例子1中的 handler 不在主线程了么
  • 参考地址

HandlerThread是Android中的一个类,它继承自Thread,主要用于在一个单独的线程中处理消息队列(MessageQueue)。以下是关于它的详细内容:

一、HandlerThread的基本概念和用途

1. 目的

  • 在Android开发中,为了避免在主线程(UI线程)执行耗时操作而导致应用程序出现“ANR(Application Not Responding)”的情况,需要将一些耗时任务(如网络请求、文件读写等)放到后台线程中执行。
  • HandlerThread提供了一种方便的方式来创建一个带有消息队列的后台线程。(和handler一起配合使用达到)
  • 它允许通过Handler发送消息到该线程的消息队列中,然后在该线程中按照消息发送的顺序依次处理这些消息。这样就可以在一个单独的线程中有序地执行一系列任务。

2. 与普通线程的区别

  • 普通线程没有自带的消息队列机制。如果要在普通线程中处理多个任务,需要自己实现任务调度和排队等复杂的逻辑。而HandlerThread内部已经实现了消息队列,并且可以通过Handler方便地与其他线程进行通信。

  • 例如,在一个普通线程中,如果要处理多个不同类型的任务,可能需要使用复杂的状态机或者阻塞队列等方式来管理任务。但是HandlerThread通过消息机制(MessageMessageQueue),可以很方便地通过sendMessage等方法发送任务请求,并且在HandlerhandleMessage方法中处理这些任务。

二、HandlerThread的使用步骤

1. 创建HandlerThread对象并启动线程

  • 首先,需要创建一个HandlerThread对象。例如:
HandlerThread handlerThread = new HandlerThread("MyHandlerThread");
handlerThread.start();
  • 这里创建了一个名为MyHandlerThreadHandlerThread,然后调用start方法来启动这个线程。启动后,该线程就会开始运行,并且创建一个与之关联的消息队列。

2. 创建Handler并关联到HandlerThread的消息队列

  • 接着,需要创建一个Handler对象,并将其与HandlerThread的消息队列关联起来。可以通过以下方式实现:
Handler handler = new Handler(handlerThread.getLooper()) {@Overridepublic void handleMessage(Message msg) {// 在这里处理消息switch (msg.what) {case 1:// 执行任务1break;case 2:// 执行任务2break;}}
};
  • 这里通过handlerThread.getLooper()获取HandlerThreadLooper对象。Looper是一个用于循环获取消息队列中的消息并分发给Handler的类。通过这种方式,创建的Handler就可以将消息发送到HandlerThread的消息队列中,并且在handleMessage方法中处理这些消息。

3. 发送消息到HandlerThread的消息队列

  • 最后,可以通过Handler发送消息到HandlerThread的消息队列中。例如:
Message message = new Message();
message.what = 1;
handler.sendMessage(message);
  • 这里创建了一个Message对象,设置了消息的what属性(用于区分不同类型的消息),然后通过handler.sendMessage方法将消息发送到HandlerThread的消息队列中。HandlerThread中的Looper会不断地从消息队列中获取消息,并将消息分发给关联的HandlerhandleMessage方法进行处理。

三、HandlerThread的生命周期和注意事项

1. 生命周期

  • HandlerThread对象被创建并调用start方法后,线程开始运行,消息队列被创建,Looper开始循环获取消息。
  • 只要还有未处理的消息在消息队列中,或者Looper没有被显式地退出,HandlerThread就会一直运行。可以通过调用HandlerThreadquit或者quitSafely方法来退出Looper,从而结束HandlerThread的运行。例如:
handlerThread.quitSafely();
  • quitSafely方法会在处理完当前消息队列中的已有消息后退出Looper,而quit方法会立即退出Looper,可能会导致消息丢失。

2. 注意事项

  • 内存泄漏:如果Handler对象是一个内部类,并且它间接引用了外部类(例如Activity)的实例,而HandlerThread的生命周期又比外部类长,那么可能会导致外部类无法被垃圾回收,从而引起内存泄漏。为了避免这种情况,可以将Handler定义为静态内部类,并使用弱引用(WeakReference)来引用外部类实例。
  • 消息处理顺序:HandlerThread中的消息是按照发送的顺序依次处理的。如果有高优先级的任务,需要在消息机制的基础上进行适当的调整,例如可以通过设置消息的优先级或者在handleMessage方法中根据任务的紧急程度优先处理某些消息。

四、使用 HandlerThread 和 线程池 举同一个例子

1. 使用HandlerThread的示例:下载文件并更新UI

  • 布局文件(activity_main.xml)
    • 简单的布局包含一个按钮用于触发下载和一个文本视图用于显示下载状态。
    <?xml version="1.0" encoding="utf-8"?>
    <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"android:orientation="vertical"android:layout_width="match_parent"android:layout_height="match_parent"><Buttonandroid:id="@+id/download_button"android:layout_width="wrap_content"android:layout_height="wrap_content"android:text="下载文件"/><TextViewandroid:id="@+id/status_text"android:layout_width="wrap_content"android:layout_height="wrap_content"/>
    </LinearLayout>
    
  • Java代码(MainActivity.java)
    • MainActivity中实现下载文件的功能。
    import android.os.Bundle;
    import android.os.Handler;
    import android.os.HandlerThread;
    import android.os.Message;
    import android.view.View;
    import android.widget.Button;
    import android.widget.TextView;
    import androidx.appcompat.app.AppCompatActivity;
    public class MainActivity extends AppCompatActivity {private HandlerThread handlerThread;private Handler handler;private TextView statusText;@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);Button downloadButton = findViewById(R.id.download_button);statusText = findViewById(R.id.status_text);// 创建HandlerThread并启动handlerThread = new HandlerThread("DownloadThread");handlerThread.start();// 创建Handler并关联到HandlerThread的消息队列handler = new Handler(handlerThread.getLooper()) {@Overridepublic void handleMessage(Message msg) {if (msg.what == 1) {// 模拟下载文件的过程try {Thread.sleep(3000);} catch (InterruptedException e) {e.printStackTrace();}// 下载完成后发送消息到主线程更新UIMessage uiMessage = new Message();uiMessage.what = 2;uiHandler.sendMessage(uiMessage);}}};downloadButton.setOnClickListener(new View.OnClickListener() {@Overridepublic void onClick(View v) {// 发送下载文件的消息到HandlerThreadMessage downloadMessage = new Message();downloadMessage.what = 1;handler.sendMessage(downloadMessage);statusText.setText("正在下载...");}});// 创建用于更新UI的主线程HandlerHandler uiHandler = new Handler(getMainLooper()) {@Overridepublic void handleMessage(Message msg) {if (msg.what == 2) {statusText.setText("下载完成");}}};}@Overrideprotected void onDestroy() {super.onDestroy();// 退出HandlerThreadhandlerThread.quitSafely();}
    }
    
    • 首先,在onCreate方法中创建HandlerThread并启动它,然后创建与HandlerThread消息队列关联的Handler。当用户点击下载按钮时,发送一个消息到HandlerThread的消息队列,在handleMessage方法中模拟文件下载过程(这里通过Thread.sleep来模拟耗时操作)。下载完成后,发送一个消息到主线程的Handler来更新UI,显示下载完成的状态。最后,在onDestroy方法中退出HandlerThread

2. 使用线程池的示例:下载文件并更新UI(同样的功能)

  • 布局文件(与上面相同,activity_main.xml)
  • Java代码(MainActivity.java)
    import android.os.Bundle;
    import android.os.Handler;
    import android.view.View;
    import android.widget.Button;
    import android.widget.TextView;
    import androidx.appcompat.app.AppCompatActivity;
    import java.util.concurrent.ExecutorService;
    import java.util.concurrent.Executors;
    public class MainActivity extends AppCompatActivity {private ExecutorService executorService;private TextView statusText;@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);Button downloadButton = findViewById(R.id.download_button);statusText = findViewById(R.id.status_text);// 创建一个单线程的线程池(这里可以根据需要调整线程池大小)executorService = Executors.newSingleThreadExecutor();downloadButton.setOnClickListener(new View.OnClickListener() {@Overridepublic void onClick(View v) {statusText.setText("正在下载...");// 提交下载任务到线程池executorService.submit(new Runnable() {@Overridepublic void run() {try {// 模拟下载文件的过程Thread.sleep(3000);} catch (InterruptedException e) {e.printStackTrace();}// 下载完成后发送消息到主线程更新UIHandler uiHandler = new Handler(getMainLooper()) {@Overridepublic void handleMessage(Message msg) {if (msg.what == 1) {statusText.setText("下载完成");}}};Message uiMessage = new Message();uiMessage.what = 1;uiHandler.sendMessage(uiMessage);}});}});}@Overrideprotected void onDestroy() {super.onDestroy();// 关闭线程池executorService.shutdown();}
    }
    
    • 这里在onCreate方法中创建一个单线程的线程池ExecutorService。当用户点击下载按钮时,将下载任务以Runnable的形式提交到线程池。在Runnablerun方法中模拟文件下载过程,下载完成后,创建一个主线程的Handler,并发送消息到主线程来更新UI,显示下载完成的状态。最后,在onDestroy方法中关闭线程池。这两种方式都实现了在后台执行下载任务,避免阻塞主线程,并且在任务完成后更新UI的功能。

3.例子1中的 handler 不在主线程了么

  1. handler不在主线程中
    • 在第一个例子中,handler关联的是HandlerThreadLooper,而HandlerThread是一个单独的后台线程。当创建handler时,通过handlerThread.getLooper()获取的是HandlerThread线程中的Looper对象。
    • 例如,代码中的这部分:
    handler = new Handler(handlerThread.getLooper()) {@Overridepublic void handleMessage(Message msg) {if (msg.what == 1) {// 模拟下载文件的过程try {Thread.sleep(3000);} catch (InterruptedException e) {e.printStackTrace();}// 下载完成后发送消息到主线程更新UIMessage uiMessage = new Message();uiMessage.what = 2;uiHandler.sendMessage(uiMessage);}}
    };
    
    • 这个handlerhandleMessage方法是在HandlerThread线程中执行的。HandlerThread提供了一个后台线程环境,这样就可以在这个线程中执行一些耗时的操作,比如模拟文件下载(通过Thread.sleep来模拟耗时),而不会阻塞主线程。
  2. 与主线程通信的uiHandler
    • 为了更新UI,又创建了一个uiHandler,它是关联到主线程(UI线程)的Looper的。
    • 代码如下:
    Handler uiHandler = new Handler(getMainLooper()) {@Overridepublic void handleMessage(Message msg) {if (msg.what == 2) {statusText.setText("下载完成");}}
    };
    
    • 当后台HandlerThread中的任务完成后,通过uiHandler发送消息到主线程的消息队列,然后在主线程中执行uiHandlerhandleMessage方法来更新UI。这是因为在Android中,只有主线程才能更新UI,所以需要这种跨线程通信的方式来在后台任务完成后更新界面显示。

参考地址

豆包 AI

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

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

css:没错又是我

背景 给元素添加背景样式 还可以设置背景颜色、背景图片&#xff08;教练我要学这个&#xff09;、背景平铺、背景图片位置、背景图像固定 背景颜色 这个我们用过&#xff0c;就是&#xff1a; a {background-color: hotpink; } 一般默认值是transparent&#xff0c;也就…

使用Git工具在GitHub的仓库中上传文件夹(超详细)

如何使用Git工具在GitHub的仓库中上传文件夹&#xff1f; 如果觉得博主写的还可以&#xff0c;点赞收藏关注噢~ 第一步&#xff1a;拥有一个本地的仓库 可以fork别人的仓库或者自己新创建 fork别人的仓库 或者自己创建一个仓库 按照要求填写完成后&#xff0c;点击按钮创建…

uniapp的基本使用(easycom规范和条件编译)和uview组件的安装和使用

文章目录 1、uniapp1.uview组件安装2.uview-plus组件安装 2、条件编译3、easycom规范1.组件路径符合规范2.自定义easycom配置的示例 总结 1、uniapp UniApp的UI组件库&#xff0c;如TMUI、uViewUI、FirstUI、TuniaoUI、ThorUI等&#xff0c;这些组件库适用于Vue3和TypeScript&…

攻防世界37-unseping-CTFWeb

攻防世界37-unseping-CTFWeb <?php highlight_file(__FILE__);class ease{private $method;private $args;function __construct($method, $args) {$this->method $method;$this->args $args;}function __destruct(){if (in_array($this->method, array("…

【大数据学习 | HBASE高级】region split机制和策略

1. region split机制 ​ HRegionServer拆分region的步骤是&#xff0c;先将该region下线&#xff0c;然后拆分&#xff0c;将其子region加入到hbase:meta表中&#xff0c;再将他们加入到原本的HRegionServer中&#xff0c;最后汇报Master。 split前&#xff1a;hbase:meta表有…

FMC 扩展子卡6 路 422,8 组 LVDS,8 路 GPIO

FMC 扩展子卡6 路 422,8 组 LVDS,8 路 GPIO 卡是一款支持多路 LVCMOS 和 LVDS 信号互转的 FMC 扩展子板。它能支持 6 路 422 信号的输入 / 输出 ,8 组 LVDS 信号的输入 / 输出和 8 路 GPIO 信号的输入 / 输出。本产品基于一些逻辑转换芯片而设计&#xff0c;能实现差分信号转单…

old-cms(原生PHP开发的企业网站管理系统)

old-cms是一个使用原生PHP开发的实用的PHP企业网站管理系统&#xff0c;包括企业网站常用的功能板块&#xff0c;如&#xff1a;产品管理、新闻管理、栏目管理、模板标签管理、分类管理、诚聘英才、在线留言反馈、关于我们&#xff08;公司简介&#xff09;等等&#xff0c;也有…

IPv4与IPv6的优缺点

IPv4 和 IPv6 都是 TCP/IP 协议的版本。IP 是指互联网协议&#xff0c;是传输控制协议/互联网协议套件&#xff08;TCP/IP&#xff09;的主要部分。 TCP/IP 是一套标准和规则&#xff0c;用于规范不同网络上的设备之间打包数据&#xff08;数据报&#xff09;的传输和交换。互…

git命令及原理

git: 目录则被称之为“树” 文件被称作 Blob 对象. git help <command>: 获取 git 命令的帮助信息 git init: 创建一个新的 git 仓库&#xff0c;其数据会存放在一个名为 .git 的目录下 git status: 显示当前的仓库状态 git add <filename>: 添加文件到暂存区 git …

aws xray通过设置采样规则对请求进行过滤

参考资料 https://github.com/aws/aws-xray-sdk-pythonpython api reference&#xff0c;https://docs.aws.amazon.com/xray-sdk-for-python/latest/reference/node api reference&#xff0c;https://docs.aws.amazon.com/xray-sdk-for-nodejs/latest/reference/ 初始化环境…

《硬件架构的艺术》笔记(一):亚稳态

同步系统中如果数据和时钟满足建立保持时间的要求&#xff0c;不会发生亚稳态&#xff08;meastable&#xff09;。 异步系统中数据和时钟关系不固定&#xff0c;可能违反建立保持时间&#xff0c;就会输出介于两个有效状态之间的中间级电平&#xff0c;且无法确定停留在中间状…

统信UOS开发环境支持Electron

全面支持Electron开发环境,同时还提供了丰富的开发工具和开发资源,进一步提升工作效率。 文章目录 一、环境部署1. Electron应用开发介绍2. Electron开发环境安装安装Node.js和npm安装electron环境配置二、代码示例Electron开发案例三、常见问题一、环境部署 1. Electron应用…

动手学深度学习68 Transformer

1. Transformer 2. 多头注意力代码 通过不断地reshape&#xff0c;避免forloop操作。 什么样的shape进去&#xff0c;怎样的shape出来。 #save class MultiHeadAttention(nn.Module):"""多头注意力"""def __init__(self, key_size, query_size…

晨控RFID技术助力半导体制造业革新之路

晨控RFID技术助力半导体制造业革新之路 应用背景 随着信息技术的快速发展&#xff0c;无线射频识别技术逐渐成为连接物理世界与数字世界的桥梁。尤其在半导体产业中&#xff0c;RFID技术的应用不仅提升了生产效率&#xff0c;还加强了产品追踪与管理能力&#xff0c;对推动产…

ReactPress:构建高效、灵活、可扩展的开源发布平台

ReactPress Github项目地址&#xff1a;https://github.com/fecommunity/reactpress 欢迎Star。 在当今数字化时代&#xff0c;内容管理系统&#xff08;CMS&#xff09;已成为各类网站和应用的核心组成部分。ReactPress&#xff0c;作为一款融合了现代Web开发多项先进技术的开…

PyTorch版本的3D网络Grad-CAM可视化实验记录

前言 最近在跑代码的时候需要可视化一些网络中间层特征来诊断网络&#xff0c;但是我的backbone是一个3D网络&#xff0c;一般的Grad-CAM都是在2D网络中应用更广泛&#xff0c;查了一下也只有几篇博文是关于3D Grad-CAM的介绍的。自己参照他们的代码试了一下&#xff0c;但是可…

基于STM32的智能充电桩:集成RTOS、MQTT与SQLite的先进管理系统设计思路

一、项目概述 随着电动车的普及&#xff0c;充电桩作为关键基础设施&#xff0c;其智能化、网络化管理显得尤为重要。本项目旨在基于STM32微控制器开发一款智能充电桩&#xff0c;能够实现高效的充电监控与管理。项目通过物联网技术&#xff0c;提供实时数据监测、远程管理、用…

.NET中通过C#实现Excel与DataTable的数据互转

在.NET框架中&#xff0c;使用C#进行Excel数据与DataTable之间的转换是数据分析、报表生成、数据迁移等操作中的常见需求。这一过程涉及到将Excel文件中的数据读取并加载至DataTable中&#xff0c;以便于利用.NET提供的丰富数据处理功能进行操作&#xff0c;同时也包括将DataTa…

域名服务系统DNS (Domain Name System)

域名的介绍 熟悉了域名之后&#xff0c;不仅仅是应对考试&#xff0c;生活中看到一个常规的网址&#xff0c;我们也能快速想到这个网址对应的含义是什么&#xff0c;并且在记忆网址的时候也更加得心应手&#xff0c;快速了解域名中各个层级的含义&#xff0c;这是 非常有趣呢…

Kettle——CSV文件转换成excel文件输出

1.点击—文件—新建—转换 拖入两个组件&#xff1a; 按shift&#xff0b;鼠标左击建立连接&#xff0c;并点击主输出步骤&#xff0c; 点击CSV文件输入&#xff0c;选择浏览的csv文件&#xff0c;然后点击确定 同样&#xff0c;Excel也同上&#xff0c;只是要删除这个xls 并…