ThreadPoolExecutor最佳实践--如何选择线程数

本文深入探讨线程池在不同任务类型(计算密集型、IO密集型及混合型)下的最优配置策略,通过实验量化分析线程数与性能的关系,揭示了线程池大小对系统资源利用与任务执行效率的影响。

去年的一篇《ThreadPoolExecutor详解》大致讲了ThreadPoolExecutor内部的代码实现。

总结一下,主要有以下四点:

  • 当有任务提交的时候,会创建核心线程去执行任务(即使有核心线程空闲仍会创建);
  • 当核心线程数达到corePoolSize时,后续提交的都会进BlockingQueue中排队;
  • 当BlockingQueue满了(offer失败),就会创建临时线程(临时线程空闲超过一定时间后,会被销毁);
  • 当线程总数达到maximumPoolSize时,后续提交的任务都会被RejectedExecutionHandler拒绝。

prestartAllCoreThreads方法可以直接创建所有核心线程并启动。

BlockingQueue使用无限容量的阻塞队列(如LinkedBlockingQueue)时,不会创建临时线程(因为队列不会满),所以线程数保持corePoolSize。

BlockingQueue使用没有容量的同步队列(如SynchronousQueue)时,任务不会入队,而是直接创建临时线程去执行任务。

线程池基本模型

虽然线程池的模型被剖析的非常清晰,但是如何最高性能地使用线程池一直是一个令人纠结的问题,其中最主要的问题就是如何决定线程池的大小

这篇文章会以量化测试的方式分析:何种情况线程池应该使用多少线程数。

1. 计算密集型任务与IO密集型任务

大多数刚接触线程池的人会认为有一个准确的值作为线程数能让线程池适用在程序的各个地方。然而大多数情况下并没有放之四海而皆准的值,很多时候我们要根据任务类型来决定线程池大小以达到最佳性能。

计算密集型任务以CPU计算为主,这个过程中会涉及到一些内存数据的存取(速度明显快于IO),执行任务时CPU处于忙碌状态。

IO密集型任务以IO为主,比如读写磁盘文件、读写数据库、网络请求等阻塞操作,执行IO操作时,CPU处于等待状态,等待过程中操作系统会把CPU时间片分给其他线程。

 

2. 计算密集型任务

下面写一个计算密集型任务的例子:

public class ComputeThreadPoolTest {

    final static ThreadPoolExecutor computeExecutor;

    final static List<Callable<Long>> computeTasks;

    final static int task_count = 5000;

    static {
        computeExecutor = (ThreadPoolExecutor) Executors.newFixedThreadPool(1);

        // 创建5000个计算任务
        computeTasks = new ArrayList<>(task_count);
        for (int i = 0; i < task_count; i++) {
            computeTasks.add(new ComputeTask());
        }
    }

    static class ComputeTask implements Callable<Long> {
        // 计算一至五十万数的总和(纯计算任务)
        @Override
        public Long call() {
            long sum = 0;
            for (long i = 0; i < 50_0000; i++) {
                sum += i;
            }
            return sum;
        }
    }

    public static void main(String[] args) throws InterruptedException {
        // 我电脑是四核处理器
        int processorsCount = Runtime.getRuntime().availableProcessors();
        // 逐一增加线程池的线程数
        for (int i = 1; i <=  processorsCount * 5; i++) {
            computeExecutor.setCorePoolSize(i);
            computeExecutor.setMaximumPoolSize(i);
            computeExecutor.prestartAllCoreThreads();
            System.out.print(i);
            computeExecutor.invokeAll(computeTasks); // warm up all thread
            System.out.print("\t");
            testExecutor(computeExecutor, computeTasks);
            System.out.println();
            // 一定要让cpu休息会儿,Windows桌面操作系统不会让应用长时间霸占CPU
            // 否则Windows回收应用程序的CPU核心数将会导致测试结果不准确
            TimeUnit.SECONDS.sleep(5);// cpu rest
        }
        computeExecutor.shutdown();
    }

    private static <T> void testExecutor(ExecutorService executor, List<Callable<T>> tasks)
        throws InterruptedException {
        for (int i = 0; i < 8; i++) {
            long start = System.currentTimeMillis();
            executor.invokeAll(tasks); // ignore result
            long end = System.currentTimeMillis();
            System.out.print(end - start); // 记录时间间隔
            System.out.print("\t");
            TimeUnit.SECONDS.sleep(1); // cpu rest
        }
    }
}
  

将程序生成的数据粘贴到excel中,并对数据进行均值统计

注意如果相同的线程数两次执行的时间相差比较大,说明测试的结果不准确。

测试生成数据

测试程序生成的数据可以从这下载

对数据生成折线图

线程池线程数与计算密集型任务执行时间关系图

由于我笔记本的CPU有四个处理器,所以会发现当线程数达到4之后,5000个任务的执行时间并没有变得更少,基本上是在600毫秒左右徘徊。

因为计算机只有四个处理器可以使用,当创建更多线程的时候,这些线程是得不到CPU的执行的。

所以对于计算密集型任务,应该将线程数设置为CPU的处理个数,可以使用Runtime.availableProcessors方法获取可用处理器的个数。

《并发编程实战》一书中对于IO密集型任务建议线程池大小设为$N_{cpu}+1$,原因是当计算密集型线程偶尔由于页缺失故障或其他原因而暂停时,这个“额外的”线程也能确保这段时间内的CPU始终周期不会被浪费

对于计算密集型任务,不要创建过多的线程,由于线程有执行栈等内存消耗,创建过多的线程不会加快计算速度,反而会消耗更多的内存空间;另一方面线程过多,频繁切换线程上下文也会影响线程池的性能。

3. 每个程序员都应该知道的延迟数

IO操作包括读写磁盘文件、读写数据库、网络请求等阻塞操作,执行这些操作,线程将处于等待状态。

为了能更准确的模拟IO操作的阻塞,我觉得有必要将https://people.eecs.berkeley.edu/~rcs/research/interactive_latency.html中列举的延迟数整理出来。

事件纳秒微秒毫秒对照
一级缓存0.5---
二级缓存7--一级缓存时间14倍
互斥锁定/解锁25.0---
主存参考100.0--二级缓存20倍,一级缓存200倍
使用Zippy压缩1K字节3,000.03--
通过1Gbps网络发送1K字节10,000.010--
从SSD中随机读取4K150,000.0150-1GB/秒的读取速度的SSD硬盘
从内存中顺序读取1MB250,000.0250--
在同一数据中心局域网内往返500,000.0500--
从SSD顺序读取1MB1,000,000.0100011GB/秒SSD,4X 内存
磁盘搜寻10,000,000.0100001020X 数据中心往返
从磁盘顺序读取1MB20,000,000.0200002080X 内存,20X SSD
发送一个数据包
美国加州→荷兰→加州
150,000,000.0150000150-

4. IO密集型任务

这里用sleep方式模拟IO阻塞:

这里用sleep方式模拟IO阻塞:
public class IOThreadPoolTest {

    // 使用无限线程数的CacheThreadPool线程池
    static ThreadPoolExecutor cachedThreadPool = (ThreadPoolExecutor) Executors.newCachedThreadPool();

    static List<Callable<Object>> tasks;

    // 仍然是5000个任务
    static int taskNum = 5000;

    static {
        tasks = new ArrayList<>(taskNum);
        for (int i = 0; i < taskNum; i++) {
            tasks.add(Executors.callable(new IOTask()));
        }
    }

    static class IOTask implements Runnable {

        @Override
        public void run() {
            try {
                TimeUnit.SECONDS.sleep(1);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
    }

    public static void main(String[] args) throws InterruptedException {
        cachedThreadPool.invokeAll(tasks);// warm up all thread
        testExecutor(cachedThreadPool, tasks);
        // 看看执行过程中创建了多少个线程
        int largestPoolSize = cachedThreadPool.getLargestPoolSize();
        System.out.println("largestPoolSize:" + largestPoolSize);
        cachedThreadPool.shutdown();
    }

    private static void testExecutor(ExecutorService executor, List<Callable<Object>> tasks)
        throws InterruptedException {
        long start = System.currentTimeMillis();
        executor.invokeAll(tasks);
        long end = System.currentTimeMillis();
        System.out.println(end - start);
    }

}
  

这里使用无线程数限制的CachedThreadPool线程池,也就是说这里的5000个任务会被5000个线程同时处理,由于所有的线程都只是阻塞而不消耗CPU资源,所以5000个任务在不到2秒的时间内就执行完了。

很明显使用CachedThreadPool能有效提高IO密集型任务的吞吐量,而且由于CachedThreadPool中的线程会在空闲60秒自动回收,所以不会消耗过多的资源。

但是打开任务管理器你会发现执行任务的同时内存会飙升到接近400M,因为每个线程都消耗了一部分内存,在5000个线程创建之后,内存消耗达到了峰值。

内存飙升

所以使用CacheThreadPool的时候应该避免提交大量长时间阻塞的任务,以防止内存溢出;另一种替代方案是,使用固定大小的线程池,并给一个较大的线程数(不会内存溢出),同时为了在空闲时节省内存资源,调用allowCoreThreadTimeOut允许核心线程超时。

线程执行栈的大小可以通过-Xsssize或-XX:ThreadStackSize参数调整

5. 混合型任务

大多数任务并不是单一的计算型或IO型,而是IO伴随计算两者混合执行的任务——即使简单的Http请求也会有请求的构造过程。

混合型任务要根据任务等待阻塞时间与CPU计算时间的比重来决定线程数量:

比如一个任务包含一次数据库读写(0.1ms),并在内存中对读取的数据进行分组过滤等操作(5μs),那么线程数应该为80左右。

线程数与阻塞比例的关系图大致如下:

线程数与阻塞比例关系图

当阻塞比例为0,也就是纯计算任务,线程数等于核心数(这里是4);阻塞比例越大,线程池的线程数应该更多

《Java并发编程实战》中最原始的公式是这样的:

$N{cpu}$代表CPU的个数,$U{cpu}$代表CPU利用率的期望值($0<U_{cpu}<1$),$\frac{W}{C}$仍然是等待时间与计算时间的比例。

我上面提供的公式相当于目标CPU利用率为100%。

通常系统中不止一个线程池,所以实际配置线程数应该将目标CPU利用率计算进去。

6. 总结

线程池的大小取决于任务的类型以及系统的特性,避免“过大”和“过小”两种极端。线程池过大,大量的线程将在相对更少的CPU和有限的内存资源上竞争,这不仅影响并发性能,还会因过高的内存消耗导致OOM;线程池过小,将导致处理器得不到充分利用,降低吞吐率。

要想正确的设置线程池大小,需要了解部署的系统中有多少个CPU,多大的内存,提交的任务是计算密集型、IO密集型还是两者兼有。

虽然线程池和JDBC连接池的目的都是对稀缺资源的重复利用,但通常一个应用只需要一个JDBC连接池,而线程池通常不止一个。如果一个系统要执行不同类型的任务,并且它们的行为差异较大,那么应该考虑使用多个线程池,使每个线程池可以根据各自的任务类型以及工作负载来调整。

 

清山绿水始于尘,博学多识贵于勤。

我有酒,你有故事吗?

微信公众号:「Java编程大本营」。

欢迎一起谈天说地,聊Java。 回复「vip课程」,获取一套价值19820 元的 java vip课程

- END

<think> 我们正在详细讨论ThreadPoolExecutor的核心线程数(corePoolSize)参数。根据用户的要求,我们需要参考之前提供的引用内容,并确保回答符合系统指令(包括LaTeX格式、引用标注等)。以下是回答的结构: 1. **定义与作用**:解释corePoolSize的基本概念和在线程池中的角色。 2. **特性详解**:包括核心线程的创建、存活机制以及特殊方法。 3. **配置建议**:结合不同任务类型(CPU密集型、IO密集型)给出设置建议。 4. **动态调整**:说明如何运行时修改corePoolSize及其影响。 5. **常见误区**:指出无界队列等配置问题。 6. **相关问题**:生成5个相关问题。 引用标注说明: - 引用[1]提到ThreadPoolExecutor是核心类,参数配置重要。 - 引用[3][4]提到核心线程数的定义。 - 引用[5]提到动态调整核心线程数的方法。 现在开始组织回答: </think> ### Java ThreadPoolExecutor 核心线程数详解 #### 一、定义与作用 **核心线程数(corePoolSize)** 是线程池中最基础的线程数量,代表线程池的**常驻处理能力**。当新任务提交时: 1. 优先由核心线程处理任务 2. 即使线程空闲也不会销毁(默认行为) 3. 构成线程池的**基线资源池**[^3][^4] 数学表达为: $$ \text{活跃线程数} = \begin{cases} \text{新建核心线程} & \text{if } \text{活跃线程} < \text{corePoolSize} \\ \text{入队等待} & \text{else if } \text{队列未满} \end{cases} $$ #### 二、核心特性详解 1. **创建机制** - 初始创建:线程池启动时默认为空(可通过 `prestartCoreThread()` 预创建) - 按需创建:新任务到达时若核心线程不足则**立即创建新线程**(即使其他核心线程空闲)[^1][^3] - *特殊方法*:`prestartAllCoreThreads()` 可提前创建所有核心线程[^5] 2. **存活机制** | 场景 | 默认行为 | 可配置行为 | |--------------------------|------------------------|------------------------------| | 无任务执行(空闲) | 永久存活 | `allowCoreThreadTimeOut(true)` 启用超时销毁[^4] | | 超时时间 | 不适用 | 与 `keepAliveTime` 参数联动 | 3. **与最大线程数的关系** - 核心线程是最大线程的子集:$0 \leq \text{corePoolSize} \leq \text{maximumPoolSize}$ - 突破核心线程数的条件: ```mermaid graph LR A[新任务提交] --> B{核心线程空闲?} B -->|是| C[由核心线程执行] B -->|否| D{核心线程数已满?} D -->|否| E[创建新核心线程] D -->|是| F[任务入队列] ``` #### 三、配置原则与实践 1. **设置依据** - **CPU密集型**:$ \text{corePoolSize} \approx \text{CPU核数} $(避免过多线程上下文切换)[^3] - **IO密集型**:$ \text{corePoolSize} \approx \text{CPU核数} \times (1 + \frac{\text{IO等待时间}}{\text{CPU计算时间}}) $ - *通用公式*:$ \text{corePoolSize} = \frac{\text{CPU核数} \times \text{目标CPU利用率} \times (1 + \frac{\text{等待时间}}{\text{计算时间}})}{1 - \text{系统预留比率}} $ 2. **动态调整(运行时修改)** ```java ThreadPoolExecutor executor = new ThreadPoolExecutor(...); executor.setCorePoolSize(8); // 动态调整核心线程数[^5] ``` - *扩容*:立即创建新线程直到达到新corePoolSize - *缩容*:超时后回收多余线程(需启用`allowCoreThreadTimeOut`) 3. **典型错误配置** - ❌ 核心线程数=0:导致每个任务都创建新线程(违背线程池初衷) - ❌ 核心线程数 >> CPU核数:引发频繁上下文切换(CPU密集型场景) - ⚠️ 无界队列+核心线程数不足:任务无限堆积导致OOM[^1][^3] #### 四、最佳实践案例 **电商秒杀场景配置** ```java int cpuCores = Runtime.getRuntime().availableProcessors(); ThreadPoolExecutor executor = new ThreadPoolExecutor( cpuCores * 2, // corePoolSize(IO密集型:核数×2) cpuCores * 8, // maxPoolSize 30, TimeUnit.SECONDS, // 非核心线程空闲30秒回收 new ArrayBlockingQueue<>(1000), // 有界队列防OOM new CustomRejectPolicy() // 自定义拒绝策略 ); ``` *设计思路*:利用核心线程承载日常流量,突发流量通过队列缓冲+非核心线程扩容[^1][^4] > **核心价值**:合理的 corePoolSize 在保障系统稳定性的同时最大化资源利用率,避免频繁创建/销毁线程的开销。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值