【Java 并发秘籍】线程池大作战:揭秘 JDK 中的线程池家族!

简介: 【8月更文挑战第24天】Java的并发库提供多种线程池以应对不同的多线程编程需求。本文通过实例介绍了四种主要线程池:固定大小线程池、可缓存线程池、单一线程线程池及定时任务线程池。固定大小线程池通过预设线程数管理任务队列;可缓存线程池能根据需要动态调整线程数量;单一线程线程池确保任务顺序执行;定时任务线程池支持周期性或延时任务调度。了解并正确选用这些线程池有助于提高程序效率和资源利用率。

Java 的并发类库为开发者提供了丰富的工具来处理多线程编程任务,其中线程池是处理并发任务的重要组成部分。线程池可以有效地管理和复用一组线程,避免频繁创建和销毁线程所带来的开销。Java 标准库中提供了多种类型的线程池实现,每种线程池都有其特定的应用场景。本文将以随笔的形式介绍 JDK 中提供的几种线程池实现,并通过示例代码展示它们的使用方法。

固定大小的线程池

固定大小的线程池通过 Executors.newFixedThreadPool(int nThreads) 方法创建。这种线程池中的线程数量是固定的,一旦创建了 n 个线程,这些线程就会一直存活,除非线程池被关闭。当有新的任务提交到线程池时,这些任务会被放入一个队列中,等待线程池中的线程空闲时来执行。

示例代码

import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;

public class FixedThreadPoolExample {
   
    public static void main(String[] args) {
   
        ExecutorService executor = Executors.newFixedThreadPool(5); // 创建一个包含5个线程的线程池

        for (int i = 0; i < 10; i++) {
   
            final int taskId = i;
            executor.submit(() -> {
   
                System.out.println("Task ID: " + taskId + " is running on thread: " + Thread.currentThread().getName());
                try {
   
                    Thread.sleep(1000); // 模拟耗时操作
                } catch (InterruptedException e) {
   
                    e.printStackTrace();
                }
            });
        }

        executor.shutdown(); // 关闭线程池
    }
}

可缓存线程池

可缓存线程池通过 Executors.newCachedThreadPool() 方法创建。这种线程池的特点是,当线程池中的线程数量超出处理任务所需要的线程数量时,多余的空闲线程会被终止。如果某个线程空闲时间超过 60 秒,那么该线程就会被终止并从线程池中移除。当需要执行新的任务时,如果线程池中有空闲的线程,就直接使用空闲线程;如果没有空闲线程,则创建新的线程。

示例代码

import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;

public class CachedThreadPoolExample {
   
    public static void main(String[] args) {
   
        ExecutorService executor = Executors.newCachedThreadPool(); // 创建一个可缓存线程池

        for (int i = 0; i < 10; i++) {
   
            final int taskId = i;
            executor.submit(() -> {
   
                System.out.println("Task ID: " + taskId + " is running on thread: " + Thread.currentThread().getName());
                try {
   
                    Thread.sleep(1000); // 模拟耗时操作
                } catch (InterruptedException e) {
   
                    e.printStackTrace();
                }
            });
        }

        executor.shutdown(); // 关闭线程池
    }
}

单一线程线程池

单一线程线程池通过 Executors.newSingleThreadExecutor() 方法创建。这种线程池只包含一个线程,因此所有任务都会按照顺序执行。这种线程池非常适合那些需要按顺序处理的任务,例如定时任务或者后台任务队列。

示例代码

import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;

public class SingleThreadExecutorExample {
   
    public static void main(String[] args) {
   
        ExecutorService executor = Executors.newSingleThreadExecutor(); // 创建一个单一线程线程池

        for (int i = 0; i < 10; i++) {
   
            final int taskId = i;
            executor.submit(() -> {
   
                System.out.println("Task ID: " + taskId + " is running on thread: " + Thread.currentThread().getName());
                try {
   
                    Thread.sleep(1000); // 模拟耗时操作
                } catch (InterruptedException e) {
   
                    e.printStackTrace();
                }
            });
        }

        executor.shutdown(); // 关闭线程池
    }
}

定时任务线程池

定时任务线程池通过 Executors.newScheduledThreadPool(int corePoolSize) 方法创建。这种线程池不仅可以执行周期性的任务,还可以在指定的延迟之后执行任务。它非常适合用来处理定时任务。

示例代码

import java.util.concurrent.Executors;
import java.util.concurrent.ScheduledExecutorService;
import java.util.concurrent.TimeUnit;

public class ScheduledThreadPoolExample {
   
    public static void main(String[] args) {
   
        ScheduledExecutorService executor = Executors.newScheduledThreadPool(5); // 创建一个包含5个线程的定时任务线程池

        Runnable task = () -> System.out.println("Executing task on thread: " + Thread.currentThread().getName());

        // 在5秒后执行任务
        executor.schedule(task, 5, TimeUnit.SECONDS);

        // 每隔3秒重复执行任务
        executor.scheduleAtFixedRate(task, 0, 3, TimeUnit.SECONDS);

        // 在5秒后开始执行任务,每隔3秒重复执行
        executor.scheduleWithFixedDelay(task, 5, 3, TimeUnit.SECONDS);

        // 关闭线程池
        executor.shutdown();
    }
}

总结

通过上述随笔,我们可以了解到 JDK 中提供了多种类型的线程池实现,每种线程池都有其特定的应用场景。选择合适的线程池类型对于优化并发程序的性能至关重要。无论是创建固定大小的线程池、可缓存线程池、单一线程线程池还是定时任务线程池,都需要根据具体的业务需求来确定。熟悉这些线程池的特点和使用方法,可以帮助我们在开发中更加高效地处理并发任务。无论是在日常开发还是面试准备中,掌握这些知识都是非常重要的。

相关文章
|
1月前
|
设计模式 缓存 安全
【JUC】(6)带你了解共享模型之 享元和不可变 模型并初步带你了解并发工具 线程池Pool,文章内还有饥饿问题、设计模式之工作线程的解决于实现
JUC专栏第六篇,本文带你了解两个共享模型:享元和不可变 模型,并初步带你了解并发工具 线程池Pool,文章中还有解决饥饿问题、设计模式之工作线程的实现
152 2
|
9月前
|
存储 监控 Java
【Java并发】【线程池】带你从0-1入门线程池
欢迎来到我的技术博客!我是一名热爱编程的开发者,梦想是编写高端CRUD应用。2025年我正在沉淀中,博客更新速度加快,期待与你一起成长。 线程池是一种复用线程资源的机制,通过预先创建一定数量的线程并管理其生命周期,避免频繁创建/销毁线程带来的性能开销。它解决了线程创建成本高、资源耗尽风险、响应速度慢和任务执行缺乏管理等问题。
453 60
【Java并发】【线程池】带你从0-1入门线程池
|
7月前
|
Java
线程池是什么?线程池在实际工作中的应用
总的来说,线程池是一种有效的多线程处理方式,它可以提高系统的性能和稳定性。在实际工作中,我们需要根据任务的特性和系统的硬件能力来合理设置线程池的大小,以达到最佳的效果。
235 18
|
10月前
|
监控 Kubernetes Java
阿里面试:5000qps访问一个500ms的接口,如何设计线程池的核心线程数、最大线程数? 需要多少台机器?
本文由40岁老架构师尼恩撰写,针对一线互联网企业的高频面试题“如何确定系统的最佳线程数”进行系统化梳理。文章详细介绍了线程池设计的三个核心步骤:理论预估、压测验证和监控调整,并结合实际案例(5000qps、500ms响应时间、4核8G机器)给出具体参数设置建议。此外,还提供了《尼恩Java面试宝典PDF》等资源,帮助读者提升技术能力,顺利通过大厂面试。关注【技术自由圈】公众号,回复“领电子书”获取更多学习资料。
|
9月前
|
安全 Java C#
Unity多线程使用(线程池)
在C#中使用线程池需引用`System.Threading`。创建单个线程时,务必在Unity程序停止前关闭线程(如使用`Thread.Abort()`),否则可能导致崩溃。示例代码展示了如何创建和管理线程,确保在线程中执行任务并在主线程中处理结果。完整代码包括线程池队列、主线程检查及线程安全的操作队列管理,确保多线程操作的稳定性和安全性。
|
1月前
|
Java
如何在Java中进行多线程编程
Java多线程编程常用方式包括:继承Thread类、实现Runnable接口、Callable接口(可返回结果)及使用线程池。推荐线程池以提升性能,避免频繁创建线程。结合同步与通信机制,可有效管理并发任务。
149 6
|
4月前
|
Java API 微服务
为什么虚拟线程将改变Java并发编程?
为什么虚拟线程将改变Java并发编程?
306 83
|
1月前
|
Java 调度 数据库
Python threading模块:多线程编程的实战指南
本文深入讲解Python多线程编程,涵盖threading模块的核心用法:线程创建、生命周期、同步机制(锁、信号量、条件变量)、线程通信(队列)、守护线程与线程池应用。结合实战案例,如多线程下载器,帮助开发者提升程序并发性能,适用于I/O密集型任务处理。
250 0
|
2月前
|
算法 Java
Java多线程编程:实现线程间数据共享机制
以上就是Java中几种主要处理多线程序列化资源以及协调各自独立运行但需相互配合以完成任务threads 的技术手段与策略。正确应用上述技术将大大增强你程序稳定性与效率同时也降低bug出现率因此深刻理解每项技术背后理论至关重要.
229 16
|
6月前
|
机器学习/深度学习 消息中间件 存储
【高薪程序员必看】万字长文拆解Java并发编程!(9-2):并发工具-线程池
🌟 ​大家好,我是摘星!​ 🌟今天为大家带来的是并发编程中的强力并发工具-线程池,废话不多说让我们直接开始。
254 0