Java 线程池预热(Warm-up)实战

 更新时间:2026年02月11日 09:38:37   作者:Knight_AL  
本文主要介绍了Java 线程池预热(Warm-up)实战,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧

一、什么是线程池预热?

线程池第一次执行任务时会:

  • 创建核心线程
  • 加载类
  • JIT 编译热点方法
  • 建立资源连接(DB / Redis / RPC)

这些都需要时间,因此第一次执行任务一般比后续慢很多

线程池预热(warm-up)就是:

✔ 在系统启动时提前创建线程
✔ 并让线程执行一次轻量任务
使其进入 已加载、已编译、已就绪 的状态。

二、Demo:不开启预热 vs 开启预热

我们做两个测试:

  1. 未预热 → 首次任务延迟明显更高
  2. 预热后 → 首次任务也能保持正常速度

三、测试代码

以下代码包含两个方法:
testWithoutWarmUp() 未预热
testWithWarmUp() 已预热

完整可运行 Demo:ThreadPoolWarmUpDemo.java

import java.util.concurrent.*;
import java.util.concurrent.atomic.AtomicBoolean;

public class ThreadPoolWarmUpDemo {

    public static void main(String[] args) throws Exception {
        System.out.println("========== 未开启预热,测试开始 ==========");
        testWithoutWarmUp();

        System.out.println("\n========== 开启预热后,测试开始 ==========");
        testWithWarmUp();
    }

    /** 用于模拟“第一次慢,后面快” */
    private static final AtomicBoolean firstRun = new AtomicBoolean(true);

    /** 模拟核心业务逻辑 */
    private static void mockBizWork() {
        try {
            if (firstRun.compareAndSet(true, false)) {
                // 只在第一次执行时慢
                Thread.sleep(50);
            } else {
                // 后续任务很快
                Thread.sleep(2);
            }
        } catch (InterruptedException e) {
            Thread.currentThread().interrupt();
        }
    }

    /** 创建一个全新的线程池(关键) */
    private static ThreadPoolExecutor newExecutor() {
        return new ThreadPoolExecutor(
                4, 4,
                60, TimeUnit.SECONDS,
                new LinkedBlockingQueue<>()
        );
    }

    /** 未开启线程池预热 */
    public static void testWithoutWarmUp() throws Exception {
        ThreadPoolExecutor executor = newExecutor();
        firstRun.set(true);

        long start = System.currentTimeMillis();

        Future<Long> future = executor.submit(() -> {
            long s = System.currentTimeMillis();
            mockBizWork();
            return System.currentTimeMillis() - s;
        });

        long firstCost = future.get();
        long totalCost = System.currentTimeMillis() - start;

        System.out.println("第一次任务耗时 = " + firstCost + " ms");
        System.out.println("整体耗时(含线程创建)= " + totalCost + " ms");

        executor.shutdown();
    }

    /** 开启预热 */
    public static void testWithWarmUp() throws Exception {
        ThreadPoolExecutor executor = newExecutor();
        firstRun.set(true);

        System.out.println(">>> 开始预热核心线程...");
        executor.prestartAllCoreThreads();

        // 预热任务:吃掉“第一次慢”
        for (int i = 0; i < 4; i++) {
            executor.submit(ThreadPoolWarmUpDemo::mockBizWork).get();
        }

        System.out.println(">>> 预热完成!");

        long start = System.currentTimeMillis();

        Future<Long> future = executor.submit(() -> {
            long s = System.currentTimeMillis();
            mockBizWork();
            return System.currentTimeMillis() - s;
        });

        long cost = future.get();
        long totalCost = System.currentTimeMillis() - start;

        System.out.println("第一次任务耗时(已预热)= " + cost + " ms");
        System.out.println("整体耗时(线程已就绪)= " + totalCost + " ms");

        executor.shutdown();
    }
}

四、对比图(示例)

场景首次任务耗时备注
❄️ 未预热55 ms包含线程创建、类加载等
🔥 已预热3 ms线程已创建、JIT 已编译,几乎“秒回”

五、什么时候必须使用线程池预热?

以下场景建议开启:

✔ 高并发系统(交易、支付、秒杀)

首个请求慢会直接影响用户体验和系统稳定性。

✔ 微服务自动扩容(K8S / Spring Cloud)

Pod 刚拉起来就被流量打满,如果没预热会导致:

  • 错误率突然增加
  • P99 延迟飙升
  • 下游熔断

✔ 接口对响应时间敏感(推荐、风控、广告)

冷启动会影响模型推理链路整体延迟。

六、总结

线程池预热很简单,但效果极其明显。

👉 不预热 = 首次任务慢几十毫秒
👉 预热后 = 首次任务几毫秒完成

通过:

executor.prestartAllCoreThreads();
executor.submit(warmTask);

即可大幅减少冷启动延迟,保证服务稳定性和低延迟能力。

到此这篇关于Java 线程池预热(Warm-up)实战的文章就介绍到这了,更多相关Java 线程池预热内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • 深入理解SpringBoot的配置环境属性

    深入理解SpringBoot的配置环境属性

    SpringBoot的配置环境属性是一种强大的工具,可以帮助我们配置和管理我们的应用程序,这篇文章主要介绍了SpringBoot的配置环境属性,需要的朋友可以参考下
    2023-07-07
  • Java的Collection集合的常用方法详解

    Java的Collection集合的常用方法详解

    这篇文章主要为大家详细介绍了Java的Collection集合的常用方法,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下,希望能够给你带来帮助
    2022-02-02
  • Java的IO流实现文件和文件夹的复制

    Java的IO流实现文件和文件夹的复制

    这篇文章主要为大家详细介绍了Java的IO流实现文件和文件夹的复制,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2022-06-06
  • java 同步器SynchronousQueue详解及实例

    java 同步器SynchronousQueue详解及实例

    这篇文章主要介绍了java 同步器SynchronousQueue详解及实例的相关资料,需要的朋友可以参考下
    2017-05-05
  • Java中getResourceAsStream用法分析

    Java中getResourceAsStream用法分析

    这篇文章主要介绍了Java中getResourceAsStream用法,较为详细的分析了getResourceAsStream的功能及用法,需要的朋友可以参考下
    2015-06-06
  • IntelliJ IDEA自定义代码提示模板Live Templates的图文教程

    IntelliJ IDEA自定义代码提示模板Live Templates的图文教程

    这篇文章主要介绍了IntelliJ IDEA自定义代码提示模板Live Templates,本文通过图文并茂的形式给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2021-03-03
  • 使用Java实现HTTP和HTTPS代理服务详解

    使用Java实现HTTP和HTTPS代理服务详解

    这篇文章主要为大家详细介绍了如何使用Java实现HTTP和HTTPS代理服务,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下
    2024-04-04
  • Windows系统下修改jar包中的依赖实现过程

    Windows系统下修改jar包中的依赖实现过程

    本文介绍了如何在Windows下修改和升级Java项目中的JAR包依赖,首先,通过解压原JAR包并替换指定的依赖JAR文件来实现升级,然后,使用命令重新打包JAR文件,整个过程包括解压、替换和打包三个步骤,确保项目能够正确引用更新的依赖
    2025-12-12
  • Idea中git的使用小结

    Idea中git的使用小结

    这篇文章主要介绍了Idea中git的使用小结,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2024-01-01
  • Mybatis通过拦截器实现单数据源内多数据库切换

    Mybatis通过拦截器实现单数据源内多数据库切换

    这篇文章主要为大家详细介绍了Mybatis如何通过拦截器实现单数据源内多数据库切换,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下
    2023-12-12

最新评论