Android 线程池实战指南:高效管理多线程任务
在 Android 开发中,线程池的使用非常重要,尤其是在需要处理大量异步任务时。线程池可以有效地管理线程资源,避免频繁创建和销毁线程带来的性能开销。以下是线程池的使用方法和最佳实践。
1. 线程池的基本使用
(1)创建线程池
Android 提供了 Executors 工厂类来创建常见的线程池,也可以通过 ThreadPoolExecutor 自定义线程池。
示例:使用 Executors 创建线程池
// 创建一个固定大小的线程池
ExecutorService fixedThreadPool = Executors.newFixedThreadPool(4);// 创建一个可缓存的线程池
ExecutorService cachedThreadPool = Executors.newCachedThreadPool();// 创建一个单线程的线程池
ExecutorService singleThreadExecutor = Executors.newSingleThreadExecutor();// 创建一个支持定时任务的线程池
ScheduledExecutorService scheduledThreadPool = Executors.newScheduledThreadPool(4);
示例:自定义线程池
int corePoolSize = Runtime.getRuntime().availableProcessors(); // 核心线程数
int maxPoolSize = corePoolSize * 2; // 最大线程数
long keepAliveTime = 30L; // 空闲线程存活时间
TimeUnit unit = TimeUnit.SECONDS; // 时间单位
BlockingQueue<Runnable> workQueue = new LinkedBlockingQueue<>(10); // 任务队列ThreadPoolExecutor customThreadPool = new ThreadPoolExecutor(corePoolSize,maxPoolSize,keepAliveTime,unit,workQueue,new ThreadPoolExecutor.CallerRunsPolicy() // 拒绝策略
);
(2)提交任务
通过 execute() 或 submit() 方法向线程池提交任务。
示例:提交任务
// 使用 execute() 提交任务
fixedThreadPool.execute(new Runnable() {@Overridepublic void run() {// 执行后台任务Log.d("ThreadPool", "Task is running on thread: " + Thread.currentThread().getName());}
});// 使用 submit() 提交任务(可以获取返回值)
Future<String> future = fixedThreadPool.submit(new Callable<String>() {@Overridepublic String call() throws Exception {// 执行后台任务并返回结果return "Task result";}
});try {String result = future.get(); // 获取任务结果Log.d("ThreadPool", "Task result: " + result);
} catch (Exception e) {e.printStackTrace();
}
(3)关闭线程池
使用完线程池后,需要调用 shutdown() 或 shutdownNow() 方法关闭线程池。
示例:关闭线程池
fixedThreadPool.shutdown(); // 平滑关闭,等待任务执行完毕
// 或者
fixedThreadPool.shutdownNow(); // 立即关闭,尝试中断正在执行的任务
2. 线程池的最佳实践
(1)根据任务类型选择线程池
FixedThreadPool:适合 CPU 密集型任务。
CachedThreadPool:适合短期异步任务。
SingleThreadExecutor:适合需要顺序执行的任务。
ScheduledThreadPool:适合定时任务或周期性任务。
(2)合理设置线程池参数
corePoolSize:根据 CPU 核心数设置。
maxPoolSize:根据任务类型设置(CPU 密集型任务设置较小,I/O 密集型任务设置较大)。
keepAliveTime:根据任务频率设置。
workQueue:根据任务数量选择合适的队列类型。
(3)避免内存泄漏
确保任务不会持有 Activity 或 Context 的引用。
在 Activity 销毁时取消线程池中的任务。
(4)处理异常
线程池中的任务如果抛出未捕获的异常,线程会终止。因此需要在任务中捕获异常。
示例:捕获异常
fixedThreadPool.execute(new Runnable() {@Overridepublic void run() {try {// 执行任务} catch (Exception e) {Log.e("ThreadPool", "Task failed: " + e.getMessage());}}
});
3. 结合 Handler 更新 UI
在 Android 中,线程池通常与 Handler 结合使用,以便将结果传递回主线程更新 UI。
示例:结合 Handler 更新 UI
Handler mainHandler = new Handler(Looper.getMainLooper());fixedThreadPool.execute(new Runnable() {@Overridepublic void run() {// 执行后台任务final String result = doBackgroundWork();// 将结果传递到主线程mainHandler.post(new Runnable() {@Overridepublic void run() {// 更新 UItextView.setText(result);}});}
});
4. 线程池的监控和调优
通过监控线程池的状态,可以及时发现性能瓶颈并进行优化。
示例:监控线程池状态
int poolSize = customThreadPool.getPoolSize(); // 当前线程池中的线程数
int activeCount = customThreadPool.getActiveCount(); // 正在执行任务的线程数
long completedTaskCount = customThreadPool.getCompletedTaskCount(); // 已完成的任务数
int queueSize = customThreadPool.getQueue().size(); // 队列中的任务数Log.d("ThreadPoolStats", "PoolSize: " + poolSize + ", ActiveCount: " + activeCount +", CompletedTaskCount: " + completedTaskCount + ", QueueSize: " + queueSize);
5. 使用第三方库
一些第三方库(如 RxJava、OkHttp 等)已经内置了线程池管理机制,可以直接使用。
示例:使用 RxJava 的线程池
Scheduler scheduler = Schedulers.from(Executors.newFixedThreadPool(4));Observable.fromCallable(() -> doBackgroundWork()).subscribeOn(scheduler).observeOn(AndroidSchedulers.mainThread()).subscribe(result -> {// 更新 UItextView.setText(result);});
6. 总结
线程池是 Android 开发中处理异步任务的重要工具。通过合理使用线程池,可以显著提升应用的性能和资源利用率。以下是关键点:
根据任务类型选择合适的线程池。
合理设置线程池参数。
避免内存泄漏和异常问题。
结合 Handler 或第三方库简化任务管理。
通过以上方法和最佳实践,你可以更好地使用线程池来优化 Android 应用的性能。
相关文章:
Android 线程池实战指南:高效管理多线程任务
在 Android 开发中,线程池的使用非常重要,尤其是在需要处理大量异步任务时。线程池可以有效地管理线程资源,避免频繁创建和销毁线程带来的性能开销。以下是线程池的使用方法和最佳实践。 1. 线程池的基本使用 (1)创建线…...
CentOS7下安装MongoDB
步骤 1:创建 MongoDB Yum 仓库文件 你需要创建一个 MongoDB 的 Yum 仓库配置文件,以便从官方源下载 MongoDB。打开终端并使用以下命令创建并编辑该文件: sudo vi /etc/yum.repos.d/mongodb-org-7.0.repo 在打开的文件中,输入以下…...
江科大51单片机笔记【15】直流电机驱动(PWM)
写在前言 此为博主自学江科大51单片机(B站)的笔记,方便后续重温知识 在后面的章节中,为了防止篇幅过长和易于查找,我把一个小节分成两部分来发,上章节主要是关于本节课的硬件介绍、电路图、原理图等理论…...
【网络协议详解】——QOS技术(学习笔记)
目录 QoS简介 QoS产生的背景 QoS服务模型 基于DiffServ模型的QoS组成 MQC简介 MQC三要素 MQC配置流程 优先级映射配置(DiffServ域模式) 优先级映射概述 优先级映射原理描述 优先级映射 PHB行为 流量监管、流量整形和接口限速简介 流量监管 流量整形 接口限速…...
【工具使用】IDEA 社区版如何创建 Spring Boot 项目(详细教程)
IDEA 社区版如何创建 Spring Boot 项目(详细教程) Spring Boot 以其简洁、高效的特性,成为 Java 开发的主流框架之一。虽然 IntelliJ IDEA 专业版提供了Spring Boot 项目向导,但 社区版(Community Edition)…...
基于Prometheus+Grafana的Deepseek性能监控实战
文章目录 1. 为什么需要专门的大模型监控?2. 技术栈组成2.1 vLLM(推理引擎层)2.2 Prometheus(监控采集层)2.3 Grafana(数据可视化平台)3. 监控系统架构4. 实施步骤4.1 启动DeepSeek-R1模型4.2 部署 Prometheus4.2.1 拉取镜像4.2.2 编写配置文件4.2.3 启动容器4.3 部署 G…...
Spring学习笔记:工厂模式与反射机制实现解耦
1.什么是Spring? spring是一个开源轻量级的java开发应用框架,可以简化企业级应用开发 轻量级 1.轻量级(对于运行环境没有额外要求) 2.代码移植性高(不需要实现额外接口) JavaEE的解决方案 Spring更像是一种解决方案,对于控制层,它有Spring…...
pytest数据库测试文章推荐
参考链接: 第一部分:http://alextechrants.blogspot.fi/2013/08/unit-testing-sqlalchemy-apps.html第二部分:http://alextechrants.blogspot.fi/2014/01/unit-testing-sqlalchemy-apps-part-2.html...
vue3 二次封装uni-ui中的组件,并且组件中有 v-model 的解决方法
在使用uniappvue3开发中, 使用了uni-ui的组件,但是我们也需要自定义组件,比如我要自定一个picker 的组件, 是在 uni-data-picker 组件的基础上进行封装的 父组件中的代码 <classesselect :selectclass"selectclass"…...
探索高性能AI识别和边缘计算 | NVIDIA Jetson Orin Nano 8GB 开发套件的全面测评
随着边缘计算和人工智能技术的迅速发展,性能强大的嵌入式AI开发板成为开发者和企业关注的焦点。NVIDIA近期推出的Jetson Orin Nano 8GB开发套件,凭借其40 TOPS算力、高效的Ampere架构GPU以及出色的边缘AI能力,引起了广泛关注。本文将从配置性…...
Prompt 工程
一、提示原則 import openai import os import openai from dotenv import load_dotenv, find_dotenv from openai import OpenAI def get_openai_key():_ load_dotenv(find_dotenv())return os.environ[OPENAI_API_KEY]client OpenAI(api_keyget_openai_key(), # This is …...
【学习笔记】《逆向工程核心原理》03.abex‘crackme-2、函数的调用约定、视频讲座-Tut.ReverseMe1
文章目录 abexcrackme-21. Visual Basic文件的特征1.1. VB专用引擎1.2. 本地代码与伪代码1.3. 事件处理程序1.4. 未文档化的结构体 2. 开始调试2.1. 间接调用2.2. RT_MainStruct结构体2.3. ThunRTMain()函数 3. 分析crackme3.1. 检索字符串3.2. 查找字符串地址3.3. 生成Serial的…...
React基础之项目实战
规范的项目结构 安装scss npm install sass -D 安装Ant Design组件库 内置了一些常用的组件 npm install antd --save 路由基础配置 npm i react-router-dom 路由基本入口 import Layout from "../page/Layout"; import Login from "../page/Login"; impor…...
SAP-ABAP:SAP数据库视图的创建图文详解
在SAP ABAP中,数据库视图(Database View)是通过ABAP字典(ABAP Dictionary)创建的。数据库视图是基于一个或多个数据库表的虚拟表,它允许你定义一种逻辑视图来访问数据。以下是创建数据库视图的步骤…...
基于深度学习的肺炎X光影像自动诊断系统实现,真实操作案例分享,值得学习!
医疗影像智能化的技术演进 医学影像分析正经历从人工判读到AI辅助诊断的革命性转变。传统放射科医师分析胸部X光片需要8-12年专业训练,而基于深度学习的智能系统可在秒级完成检测。本文将以肺炎X光检测为切入点,详解从数据预处理到模型部署的全流程实现。…...
Unity Shader学习总结
1.帧缓冲区和颜色缓冲区区别 用于存储每帧每个像素颜色信息的缓冲区 帧缓冲区包括:颜色缓冲区 深度缓冲区 模板缓冲区 自定义缓冲区 2.ImageEffectShader是什么 后处理用的shader模版 3.computerShader 独立于渲染管线之外,在显卡上运行,大量…...
算法精讲 | 树(番外):平衡世界的四大守护者:AVL vs 红黑树 vs B树 vs B+树
🌲 算法精讲 | 树(番外):平衡世界的四大守护者:AVL vs 红黑树 vs B树 vs B树 📅 2025/03/12 || 🌟 推荐阅读时间 30分钟 🚀 开篇:数据结构界的四大天王 想象你是一名图书…...
第十八:go 并发 goroutine
channel 可以让多个goroutine 之间实现通信 Add方法调用时机:必须在goroutine 启动之前调用Add方法来增加计数器的值。 如果在goroutine已经启动之后再调用Add,可能会导致Wait方法提前返回,因为计数器没有正确反映正在运行的goroutine的数量…...
在vs中无法用QtDesigner打开ui文件的解决方法
解决方法 右键ui文件,选择打开方式,弹出如下界面。 点击添加,弹出如下界面 点击程序后边的三个点,去电脑查找designer.exe,我的位置为D:\Qt\Qt5.9.9\5.9.9\msvc2015_64\bin\designer.exe。 名称可以自己起一个名字,…...
【Maven教程与实战案例】
文章目录 前言一、Maven是什么?二、Maven的安装与配置1. 安装前置条件2. 下载与配置 Maven3. 验证安装 三、Maven的核心概念1. POM.xml 文件2. 构建生命周期与插件机制 四、实战项目示例1. 项目目录结构2. 编写代码App.javaAppTest.java 3. 构建项目4. 运行项目 前言…...
基于SSM的海外代购系统
一、 项目介绍 基于SSM的海外代购系统 角色:管理员、用户、代购员 管理员: 管理员登录海外代购系统可以添加、修改或者删除首页、代购员、用户、商品分类、海外代购、采购入库、系统管理、订单管理、用户资料 等。 用户:当用户打开系统的网…...
图像识别技术与应用-YOLO
1 YOLO-V1 YOLO-V1它是经典的one-stage方法,You Only Look Once,名字就已经说明了一切!把检测问题转化成回归问题,一个CNN就搞定了!也可以对视频进行实时检测,应用领域非常广! YOLO-V1诞生与2…...
严格把控K8S集群中的操作权限,为普通用户生成特定的kubeconfig文件
文章目录 前言一、背景二、证书和证书签名请求(了解)1.证书签名请求2.请求签名流程3.Kubernetes 签名者4.证书过期时间限制字段 二、脚本示例2.检查集群上下文及csr3.切换集群上下文,检查权限4.普通用户操作 总结 前言 使用并维护过K8S的ops/sre都知道,kubeconfig对于k8s的访问…...
LLM推理和优化(1):基本概念介绍
一、LLM推理的核心过程:自回归生成 LLM(如DeepSeek、ChatGPT、LLaMA系列等)的推理本质是自回归生成:从初始输入(如[CLS]或用户prompt)开始,逐token预测下一个词,直到生成结束符&…...
Kubernetes教程(七)了解集群、标签、Pod和Deployment
了解集群、标签、Pod和Deployment 一、K8s资源对象二、K8s集群1. Master2. Node 三、Namespace(命名空间)四、Label(标签)五、Pod1. 共享网络命名空间2. 共享数据 六、工作负载1. 设置副本数2. 应用升级 结语 Kubernetes的知识真的…...
zerotier搭建免费moon服务器
🌟 前言 ZeroTier是一种基于P2P的虚拟组网工具,通过搭建Moon服务器可大幅提升跨运营商/跨国节点的连接质量。本文使用云服务演示部署流程。 📋 准备工作 注册三丰云账号 创建CentOS 8.5实例 (这里选择centos8以上&a…...
【网络安全 | 漏洞挖掘】四链路账户接管
未经许可,不得转载。 文章目录 正文正文 这一过程始于身份验证流程中的 IDOR 漏洞。登录时,后台会发送多个请求。在 Burp Suite 分析这些请求时,我注意到一个值得关注的请求: 请求: POST /validateUser {"email": "victim@example.com" }响应: {…...
【最新】DeepSeek 实用集成工具有那些?
deepseek 系列github仓库地址 【主页】deepseek-aiDeepSeek-R1DeepSeek-V3DeepSeek-VL2【本文重点介绍】awesome-deepseek-integration 注意:以下内容来自awesome-deepseek-integration DeepSeek 实用集成(awesome-deepseek-integration) 将…...
linux 的免密切换用户PAM配置
/etc/pam.d/su是Linux系统中与用户切换(su命令)相关的PAM(Pluggable Authentication Modules,可插拔认证模块)配置文件。以下是对它的详细介绍: 简介 作用 PAM是一种用于管理系统认证的机制,…...
Flutter_学习记录_video_player、chewie 播放视频
1. video_player 视频播放 插件地址:https://pub.dev/packages/video_player 添加插件 导入头文件 import package:video_player/video_player.dart;Android配置(iOS不用配置) 修改这个文件:/android/app/src/main/AndroidMani…...
