Java并发:ExecutorService控制线程数技巧
文章不知道大家是否熟悉?今天我将给大家介绍《Java并发:ExecutorService控制线程数方法》,这篇文章主要会讲到等等知识点,如果你在看完本篇文章后,有更好的建议或者发现哪里有问题,希望大家都能积极评论指出,谢谢!希望我们能一起加油进步!

本教程详细介绍了如何在Java应用程序中利用`Executors`框架来限制并发执行的线程数量。通过创建固定大小的线程池(`FixedThreadPool`),您可以有效地管理任务的并行执行,避免资源过度消耗。文章将涵盖任务的定义(`Runnable`或`Callable`)、线程池的创建与任务提交,以及确保线程池优雅关闭的关键步骤,并提供实际代码示例。
1. 理解并发任务与线程池的必要性
在开发高性能的Java应用程序时,经常会遇到需要同时处理大量独立任务的场景,例如批量数据处理、文件I/O操作或网络请求。直接为每个任务创建一个新线程虽然简单,但会导致线程数量失控,进而引发系统资源耗尽(如内存溢出)、上下文切换开销增大,甚至程序崩溃。
Java 5引入的Executors框架提供了一种更高级、更健壮的方式来管理线程,即通过线程池(ThreadPoolExecutor)。线程池允许我们预先创建一组线程,并重复利用这些线程来执行任务,从而有效限制并发量,提高资源利用率和系统稳定性。
2. 定义并发任务:Runnable与Callable
在Java中,并发任务通常通过实现Runnable或Callable接口来定义。
- Runnable: 适用于不需要返回结果的任务。它的run()方法不接受参数,也不返回任何值。
- Callable: 适用于需要返回结果或可能抛出异常的任务。它的call()方法可以返回一个泛型结果,并声明抛出异常。
为了演示,我们假设有一个数据序列化任务。我们将把原始问题中的serializeDestinationEmploye方法封装到一个Runnable任务中。
import com.google.gson.Gson;
import java.io.FileWriter;
import java.io.IOException;
import java.io.Writer;
import java.nio.file.Path; // 假设有一个路径来存储文件
// 假设 EventuelleDestination, EmployeDao, EntrepriseDao 是已定义的类
// 这里为了简化,我们只关注序列化逻辑
class EventuelleDestination {
private String id; // 假设有一个ID
private String name; // 假设有一个名称
// ... 其他属性和getter/setter
public EventuelleDestination(String id, String name) {
this.id = id;
this.name = name;
}
public String getId() { return id; }
public String getName() { return name; }
@Override
public String toString() { return "EventuelleDestination{" + "id='" + id + '\'' + ", name='" + name + '\'' + '}'; }
}
class SerializationTask implements Runnable {
private final EventuelleDestination destination;
private final Path outputDirectory; // 输出目录
public SerializationTask(EventuelleDestination destination, Path outputDirectory) {
this.destination = destination;
this.outputDirectory = outputDirectory;
}
@Override
public void run() {
// 模拟原始的序列化逻辑
// 实际应用中,employeDao和entrepriseDao应通过依赖注入或其他方式获取
// 这里为了演示,我们简化文件名生成
String filename = "/" + destination.getId() + "_" + destination.getName() + ".json";
try (Writer writer = new FileWriter(outputDirectory.resolve(filename).toFile())) {
new Gson().toJson(destination, writer);
System.out.println(Thread.currentThread().getName() + " - " + destination + " has been serialized...");
} catch (IOException e) {
System.err.println(Thread.currentThread().getName() + " - Error serializing " + destination + ": " + e.getMessage());
e.printStackTrace();
}
}
}在上述代码中,SerializationTask实现了Runnable接口,其run方法包含了对单个EventuelleDestination对象进行JSON序列化的逻辑。通过在输出中包含Thread.currentThread().getName(),我们可以观察到是哪个线程执行了任务。
3. 创建并管理固定大小的线程池
为了限制并发线程的数量,Java的Executors框架提供了Executors.newFixedThreadPool(int nThreads)方法。这个方法会创建一个拥有固定线程数量的ExecutorService。当提交的任务多于线程池中的线程数时,多余的任务会在队列中等待,直到有空闲线程可用。
以下是如何创建固定线程池并提交任务的示例:
import java.nio.file.Files;
import java.nio.file.Path;
import java.nio.file.Paths;
import java.util.ArrayList;
import java.util.List;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.TimeUnit;
public class ThreadPoolDemo {
public static void main(String[] args) {
ThreadPoolDemo demo = new ThreadPoolDemo();
demo.runSerializationDemo();
}
private void runSerializationDemo() {
// 1. 准备任务数据
List<EventuelleDestination> destinations = new ArrayList<>();
for (int i = 0; i < 20; i++) {
destinations.add(new EventuelleDestination("ID_" + i, "Location_" + i));
}
// 2. 指定输出目录
Path outputDir = Paths.get("serialized_data");
try {
Files.createDirectories(outputDir); // 确保目录存在
} catch (IOException e) {
System.err.println("Failed to create output directory: " + e.getMessage());
return;
}
// 3. 创建固定大小的线程池,限制同时运行3个线程
// 这里的3是根据原始问题中的需求设定的
ExecutorService executorService = Executors.newFixedThreadPool(3);
System.out.println("ExecutorService created with 3 threads.");
// 4. 提交所有任务到线程池
for (EventuelleDestination dest : destinations) {
executorService.submit(new SerializationTask(dest, outputDir));
}
// 5. 优雅地关闭线程池
shutdownAndAwaitTermination(executorService);
System.out.println("All serialization tasks completed.");
}
/**
* 优雅地关闭ExecutorService,等待所有任务完成。
* 这是来自Javadoc的推荐模式。
* @param executorService 要关闭的ExecutorService
*/
void shutdownAndAwaitTermination(ExecutorService executorService) {
executorService.shutdown(); // 禁用新任务提交
try {
// 等待现有任务在指定时间内终止
if (!executorService.awaitTermination(60, TimeUnit.SECONDS)) {
executorService.shutdownNow(); // 如果超时,取消当前正在执行的任务
// 再次等待,确保任务对取消做出响应
if (!executorService.awaitTermination(60, TimeUnit.SECONDS)) {
System.err.println("ExecutorService did not terminate.");
}
}
} catch (InterruptedException ie) {
// (重新)取消如果当前线程也被中断
executorService.shutdownNow();
// 保留中断状态
Thread.currentThread().interrupt();
}
}
}在runSerializationDemo方法中:
- 我们创建了一个EventuelleDestination对象列表,模拟待处理的数据。
- 指定了一个用于存储序列化文件的输出目录。
- 通过Executors.newFixedThreadPool(3)创建了一个只包含3个线程的线程池。这意味着无论我们提交多少任务,最多只有3个任务会同时运行。
- 将所有SerializationTask实例提交给executorService。submit()方法会将任务放入线程池的内部队列,并由线程池中的空闲线程按序执行。
- 最后,调用shutdownAndAwaitTermination方法来确保线程池被正确关闭。
4. 优雅地关闭ExecutorService
正确关闭ExecutorService至关重要,以防止资源泄露或程序无法退出。ExecutorService提供了两种关闭机制:
- shutdown(): 启动有序关闭,不再接受新任务,但会完成所有已提交的任务(包括正在执行的和在队列中等待的)。
- shutdownNow(): 尝试立即停止所有正在执行的任务,并清空任务队列中所有未开始的任务。它会返回一个未执行任务的列表。
通常,最佳实践是先调用shutdown(),然后使用awaitTermination()等待一段时间,让已提交的任务有机会完成。如果超时仍未完成,则可以调用shutdownNow()强制关闭。上述shutdownAndAwaitTermination方法演示了这种优雅关闭的模式。
5. 注意事项与最佳实践
- 线程池大小的考量: 固定线程池的大小应根据应用程序的特性和系统资源来决定。
- CPU密集型任务: 线程数通常设置为CPU核心数或核心数+1,以避免过多的上下文切换。
- I/O密集型任务: 线程数可以适当调大,因为线程在等待I/O操作时不会占用CPU,其他线程可以利用CPU。一个常见的经验法则是 CPU核心数 * (1 + 等待时间/计算时间)。
- 异常处理: 在Runnable或Callable的run()/call()方法中,务必包含健壮的异常处理逻辑,以防止单个任务的失败导致整个线程崩溃或影响其他任务。
- 任务结果处理(针对Callable): 如果使用Callable,executorService.submit()会返回一个Future对象。您可以使用Future.get()方法来获取任务的执行结果,或者检查任务是否完成以及是否抛出异常。
- 日志与监控: 在多线程环境中,日志输出的顺序可能与实际执行顺序不一致。在日志中包含线程名称或时间戳(如Instant.now())有助于调试和分析。同时,监控线程池的状态(如活动线程数、队列大小)对于诊断性能问题至关重要。
- 避免无限期等待: 在awaitTermination中使用超时时间是防止程序无限期阻塞的关键。
总结
通过Java的Executors框架,特别是Executors.newFixedThreadPool(),我们可以轻松地实现对并发线程数量的精确控制。这种方法不仅简化了多线程编程,还提高了应用程序的健壮性、资源利用率和可维护性。理解任务的定义、线程池的生命周期管理以及优雅关闭机制,是编写高效、稳定并发程序的基石。
终于介绍完啦!小伙伴们,这篇关于《Java并发:ExecutorService控制线程数技巧》的介绍应该让你收获多多了吧!欢迎大家收藏或分享给更多需要学习的朋友吧~golang学习网公众号也会发布文章相关知识,快来关注吧!
京东预售尾款怎么付?预售不能合并付款原因解析
- 上一篇
- 京东预售尾款怎么付?预售不能合并付款原因解析
- 下一篇
- 退出账号登录方法详解
-
- 文章 · java教程 | 7分钟前 |
- SpringMVC表单绑定优化方法
- 473浏览 收藏
-
- 文章 · java教程 | 23分钟前 | java 开发工具
- MacOSJava开发环境配置教程
- 311浏览 收藏
-
- 文章 · java教程 | 40分钟前 |
- 本地快速搭建Java环境教程
- 190浏览 收藏
-
- 文章 · java教程 | 1小时前 |
- JIT编译器优化提升Java性能方法
- 486浏览 收藏
-
- 文章 · java教程 | 1小时前 |
- Java定时器使用技巧与API解析
- 413浏览 收藏
-
- 文章 · java教程 | 1小时前 |
- Javaimport作用与加载机制解析
- 363浏览 收藏
-
- 文章 · java教程 | 1小时前 |
- 设置Java应用启动参数的Windows方法
- 202浏览 收藏
-
- 文章 · java教程 | 1小时前 |
- Java中double类型使用方法详解
- 292浏览 收藏
-
- 文章 · java教程 | 1小时前 |
- Java链表反转内存溢出解决方法
- 243浏览 收藏
-
- 文章 · java教程 | 2小时前 |
- Java图书推荐系统:筛选与遍历实战教程
- 431浏览 收藏
-
- 文章 · java教程 | 2小时前 | java rmi
- RMI远程调用原理及实现解析
- 254浏览 收藏
-
- 文章 · java教程 | 3小时前 |
- 构造代码块与静态代码块区别解析
- 200浏览 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 543次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 516次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 500次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 485次学习
-
- ChatExcel酷表
- ChatExcel酷表是由北京大学团队打造的Excel聊天机器人,用自然语言操控表格,简化数据处理,告别繁琐操作,提升工作效率!适用于学生、上班族及政府人员。
- 3485次使用
-
- Any绘本
- 探索Any绘本(anypicturebook.com/zh),一款开源免费的AI绘本创作工具,基于Google Gemini与Flux AI模型,让您轻松创作个性化绘本。适用于家庭、教育、创作等多种场景,零门槛,高自由度,技术透明,本地可控。
- 3708次使用
-
- 可赞AI
- 可赞AI,AI驱动的办公可视化智能工具,助您轻松实现文本与可视化元素高效转化。无论是智能文档生成、多格式文本解析,还是一键生成专业图表、脑图、知识卡片,可赞AI都能让信息处理更清晰高效。覆盖数据汇报、会议纪要、内容营销等全场景,大幅提升办公效率,降低专业门槛,是您提升工作效率的得力助手。
- 3710次使用
-
- 星月写作
- 星月写作是国内首款聚焦中文网络小说创作的AI辅助工具,解决网文作者从构思到变现的全流程痛点。AI扫榜、专属模板、全链路适配,助力新人快速上手,资深作者效率倍增。
- 4855次使用
-
- MagicLight
- MagicLight.ai是全球首款叙事驱动型AI动画视频创作平台,专注于解决从故事想法到完整动画的全流程痛点。它通过自研AI模型,保障角色、风格、场景高度一致性,让零动画经验者也能高效产出专业级叙事内容。广泛适用于独立创作者、动画工作室、教育机构及企业营销,助您轻松实现创意落地与商业化。
- 4083次使用
-
- 提升Java功能开发效率的有力工具:微服务架构
- 2023-10-06 501浏览
-
- 掌握Java海康SDK二次开发的必备技巧
- 2023-10-01 501浏览
-
- 如何使用java实现桶排序算法
- 2023-10-03 501浏览
-
- Java开发实战经验:如何优化开发逻辑
- 2023-10-31 501浏览
-
- 如何使用Java中的Math.max()方法比较两个数的大小?
- 2023-11-18 501浏览

