Android HandlerThread 基础

news2024/11/8 6:08:32

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()) {
    @Override
    public void handleMessage(Message msg) {
        // 在这里处理消息
        switch (msg.what) {
            case 1:
                // 执行任务1
                break;
            case 2:
                // 执行任务2
                break;
        }
    }
};
  • 这里通过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">
        <Button
            android:id="@+id/download_button"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:text="下载文件"/>
        <TextView
            android: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;
        @Override
        protected 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()) {
                @Override
                public void handleMessage(Message msg) {
                    if (msg.what == 1) {
                        // 模拟下载文件的过程
                        try {
                            Thread.sleep(3000);
                        } catch (InterruptedException e) {
                            e.printStackTrace();
                        }
                        // 下载完成后发送消息到主线程更新UI
                        Message uiMessage = new Message();
                        uiMessage.what = 2;
                        uiHandler.sendMessage(uiMessage);
                    }
                }
            };
            downloadButton.setOnClickListener(new View.OnClickListener() {
                @Override
                public void onClick(View v) {
                    // 发送下载文件的消息到HandlerThread
                    Message downloadMessage = new Message();
                    downloadMessage.what = 1;
                    handler.sendMessage(downloadMessage);
                    statusText.setText("正在下载...");
                }
            });
            // 创建用于更新UI的主线程Handler
            Handler uiHandler = new Handler(getMainLooper()) {
                @Override
                public void handleMessage(Message msg) {
                    if (msg.what == 2) {
                        statusText.setText("下载完成");
                    }
                }
            };
        }
        @Override
        protected void onDestroy() {
            super.onDestroy();
            // 退出HandlerThread
            handlerThread.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;
        @Override
        protected 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() {
                @Override
                public void onClick(View v) {
                    statusText.setText("正在下载...");
                    // 提交下载任务到线程池
                    executorService.submit(new Runnable() {
                        @Override
                        public void run() {
                            try {
                                // 模拟下载文件的过程
                                Thread.sleep(3000);
                            } catch (InterruptedException e) {
                                e.printStackTrace();
                            }
                            // 下载完成后发送消息到主线程更新UI
                            Handler uiHandler = new Handler(getMainLooper()) {
                                @Override
                                public void handleMessage(Message msg) {
                                    if (msg.what == 1) {
                                        statusText.setText("下载完成");
                                    }
                                }
                            };
                            Message uiMessage = new Message();
                            uiMessage.what = 1;
                            uiHandler.sendMessage(uiMessage);
                        }
                    });
                }
            });
        }
        @Override
        protected 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()) {
        @Override
        public void handleMessage(Message msg) {
            if (msg.what == 1) {
                // 模拟下载文件的过程
                try {
                    Thread.sleep(3000);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
                // 下载完成后发送消息到主线程更新UI
                Message 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()) {
        @Override
        public void handleMessage(Message msg) {
            if (msg.what == 2) {
                statusText.setText("下载完成");
            }
        }
    };
    
    • 当后台HandlerThread中的任务完成后,通过uiHandler发送消息到主线程的消息队列,然后在主线程中执行uiHandlerhandleMessage方法来更新UI。这是因为在Android中,只有主线程才能更新UI,所以需要这种跨线程通信的方式来在后台任务完成后更新界面显示。

参考地址

豆包 AI

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

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

相关文章

正则表达式在Kotlin中的应用:提取图片链接

在现代的Web开发中&#xff0c;经常需要从网页内容中提取特定的数据&#xff0c;例如图片链接。Kotlin作为一种现代的编程语言&#xff0c;提供了强大的网络请求和文本处理能力。本文将介绍如何使用Kotlin结合正则表达式来提取网页中的图片链接。 正则表达式基础 正则表达式是…

鉴源实验室·加密技术在汽车系统中的应用

随着汽车技术的快速发展&#xff0c;现代汽车已经不再是简单的交通工具&#xff0c;而是融合了多种智能功能的移动终端。无论是自动驾驶、车联网&#xff08;V2X&#xff09;&#xff0c;还是车内娱乐系统&#xff0c;数据传输和存储已经成为汽车生态系统中的关键环节。然而&am…

UE5.1 控制台设置帧率

仅个人记录&#xff0c;未经过严格验证。 也可通过控制台命令蓝图节点&#xff0c;在运行时执行 锁帧&#xff1a; 0->120帧 1-》60帧

SpringCloud Sentinel 服务治理详解

雪崩问题 微服务调用链路中的某个服务故障&#xff0c;引起整个链路中的所有微服务都不可用&#xff0c;这就是雪崩。 雪崩问题产生的原因&#xff1a; 微服务相互调用&#xff0c;服务提供者出现故障或阻塞。服务调用者没有做好异常处理&#xff0c;导致自身故障。调用链中的…

前端基础-html-注册界面

&#xff08;200粉啦&#xff0c;感谢大家的关注~ 一起加油吧~&#xff09; 浅浅分享下作业&#xff0c;大佬轻喷~ 网页最终效果&#xff1a; 详细代码&#xff1a; ​ <!DOCTYPE html> <html lang"en"><head><meta charset"UTF-8"…

《TCP/IP网络编程》学习笔记 | Chapter 4:基于TCP的服务器端/客户端(2)

《TCP/IP网络编程》学习笔记 | Chapter 4&#xff1a;基于TCP的服务器端/客户端&#xff08;2&#xff09; 《TCP/IP网络编程》学习笔记 | Chapter 4&#xff1a;基于TCP的服务器端/客户端&#xff08;2&#xff09;回声客户端的完美实现回声客户端的问题回声客户端问题的解决方…

使用 FFmpeg 进行音视频转换的相关命令行参数解释

FFmpeg 是一个强大的多媒体框架&#xff0c;能够解码、编码、转码、录制、播放以及流化几乎所有类型的音频和视频。它广泛应用于音视频处理任务中&#xff0c;包括格式转换、剪辑、合并、水印添加等。本文中简鹿办公将介绍如何使用 FFmpeg 进行一些常见的音视频转换任务。 安装…

ctfshow(316)--XSS漏洞--反射性XSS

Web316 进入界面&#xff1a; 审计 显示是关于反射性XSS的题目。 思路 首先想到利用XSS平台解题&#xff0c;看其他师傅的wp提示flag是在cookie中。 当前页面的cookie是flagyou%20are%20not%20admin%20no%20flag。 但是这里我使用XSS平台&#xff0c;显示的cookie还是这样…

从0开始学习Linux——网络配置

往期目录&#xff1a; 从0开始学习Linux——简介&安装 从0开始学习Linux——搭建属于自己的Linux虚拟机 从0开始学习Linux——文本编辑器 从0开始学习Linux——Yum工具 从0开始学习Linux——远程连接工具 从0开始学习Linux——文件目录 上一个教程中&#xff0c;我们了解了…

python在word中插入图片

本文讲解python如何在word文档中插入图片&#xff0c;以及指定插入图片的段落。 1、在新建的word文档中插入图片 import win32com.client as win32 from win32com.client import constants # 1&#xff09;打开word应用程序 doc_app win32.gencache.EnsureDispatch(Word.App…

亚信安全新一代WAF:抵御勒索攻击的坚固防线

近年来&#xff0c;勒索攻击已成为黑客的主要攻击手段。新型勒索攻击事件层出不穷&#xff0c;勒索攻击形势愈发严峻&#xff0c;已经对全球制造、金融、能源、医疗、政府组织等关键领域造成严重危害。如今&#xff0c;勒索攻击手段日趋成熟、攻击目标愈发明确&#xff0c;模式…

Linux qt下是使用搜狗輸入發

1.下载一个编译好的包 https://github.com/sixsixQAQ/fcitx5-qt 出处&#xff1a;这里 2.根据QT5&#xff0c;或者QT6选择下载 3.使用 把那个libfcitx5platforminputcontextplugin.so放到下面的路径&#xff1a; <你的Qt安装目录>/gcc_64/plugins/platforminputcontex…

linux命令详解,账号相关

账号相关 用户账号数据库相关文件 /etc/password 存储系统中所有用户账户的基本信息 /etc/shadow 用于存储用户账户的密码和其他安全相关信息 /etc/gshdow 用于存储用户组的密码和其他安全相关信息/etc/passwd: username:password:UID:GID:GECOS:home_directory:shell字段解…

Sentinel — 微服务保护

微服务架构将大型应用程序拆分为多个小而独立的服务&#xff0c;每个服务可以独立部署和扩展。然而&#xff0c;微服务系统需要面对的挑战也随之增加&#xff0c;例如服务之间的依赖、分布式环境下的故障传播和安全问题。因此&#xff0c;微服务保护措施是确保系统在高并发、资…

使用Qt制作一个流程变更申请流程进度以及未读消息提醒

1.1加载界面&#xff1a; 界面要素&#xff1a; 成员信息 变更位置申请 接受消息列表 根据角色加载对应界面。 1.2发起变更申请&#xff1a; 用户点击“发起变更申请”按钮。变更申请对话框可编辑&#xff0c;用户填写申请信息&#xff1a; 申请方&#xff08;自动填充&…

Markdown 全面教程:从基础到高级

Markdown 全面教程&#xff1a;从基础到高级 Markdown 是一种轻量级的标记语言&#xff0c;它的设计目标是使书写和阅读文档变得简单而直观。无论是撰写博客、编写文档还是创建 README 文件&#xff0c;Markdown 都是一个非常实用的工具。 目录 Markdown 简介Markdown 的基…

无插件H5播放器EasyPlayer.js关于硬解码和软解码的详细介绍

在当今这个多媒体内容日益丰富的时代&#xff0c;视频播放体验的重要性不言而喻.EasyPlayer.js H5播放器作为一款专为现代Web环境设计的播放器&#xff0c;它不仅提供了流畅的播放体验&#xff0c;还特别注重性能优化。EasyPlayer.js支持多种解码方式&#xff0c;包括硬解码和软…

Multi‐modal knowledge graph inference via media convergenceand logic rule

摘要 媒体融合通过处理来自不同模式的信息并将其应用于不同的领域来实现。传统的知识图很难利用多媒体特征&#xff0c;因为从其他模态引入大量信息降低了表示学习的有效性&#xff0c;并降低了知识图推理的有效性。为了解决这一问题&#xff0c;提出了一种基于媒体融合和规则…

大模型应用编排工具Dify二开之工具和模型页面改造

1.前言 简要介绍下 dify&#xff1a; ​ 一款可以对接市面上主流大模型的任务编排工具&#xff0c;可以通过拖拽形式进行编排形成解决某些业务场景的大模型应用。 背景信息&#xff1a; ​ 环境&#xff1a;dify-0.8.3、docker-21 ​ 最近笔者在做 dify的私有化部署和二次…

【数学】通用三阶矩阵特征向量的快速求法 超简单!!!

目录 三个定理1、3个特征值&#xff08;即根互不相等&#xff09;例题实践2、2个特征值&#xff08;即有一个双重根&#xff09;3、1个特征值&#xff08;即有一个三重根&#xff09;定理证明 三个定理 本定理适用于 所有三阶矩阵 的特征向量求法&#xff01; 1、3个特征值&…