在 Spring Boot 的 Web 容器中,完全可以维护多个线程池,这是一种非常常见的架构设计。不同业务场景对线程模型的需求不同,通过隔离线程池可以避免相互影响,提升系统的稳定性和可观测性。
以下是 Web 容器内常见的几种线程池及其用途:
1. 通用业务线程池
处理非阻塞、可异步化的业务逻辑。
java
@Configuration
public class ThreadPoolConfig {
@Bean("businessExecutor")
public ThreadPoolTaskExecutor businessExecutor() {
ThreadPoolTaskExecutor executor = new ThreadPoolTaskExecutor();
executor.setCorePoolSize(10);
executor.setMaxPoolSize(20);
executor.setQueueCapacity(100);
executor.setThreadNamePrefix("business-");
executor.setRejectedExecutionHandler(new ThreadPoolExecutor.CallerRunsPolicy());
executor.initialize();
return executor;
}
}
2. IO 密集型线程池
适用于大量 IO 操作(数据库查询、HTTP 调用、文件读写等),通常设置较大的线程数。
java
@Bean("ioExecutor")
public ThreadPoolTaskExecutor ioExecutor() {
ThreadPoolTaskExecutor executor = new ThreadPoolTaskExecutor();
// 线程数可设置为 CPU 核心数 * 2 或更高
executor.setCorePoolSize(50);
executor.setMaxPoolSize(100);
executor.setQueueCapacity(500);
executor.setThreadNamePrefix("io-");
return executor;
}
3. CPU 密集型线程池
适用于计算密集型任务,线程数通常设置为 CPU 核心数。
java
@Bean("cpuExecutor")
public ThreadPoolTaskExecutor cpuExecutor() {
ThreadPoolTaskExecutor executor = new ThreadPoolTaskExecutor();
int processors = Runtime.getRuntime().availableProcessors();
executor.setCorePoolSize(processors);
executor.setMaxPoolSize(processors);
executor.setThreadNamePrefix("cpu-");
return executor;
}
4. 定时任务线程池
处理 @Scheduled 注解的定时任务,可以独立配置。
java
@Bean("scheduledExecutor")
public ThreadPoolTaskScheduler scheduledExecutor() {
ThreadPoolTaskScheduler scheduler = new ThreadPoolTaskScheduler();
scheduler.setPoolSize(10);
scheduler.setThreadNamePrefix("scheduled-");
return scheduler;
}
5. 异步事件处理线程池
用于 @Async 异步方法或应用事件监听器的处理。
java
@Bean("asyncExecutor")
public ThreadPoolTaskExecutor asyncExecutor() {
ThreadPoolTaskExecutor executor = new ThreadPoolTaskExecutor();
executor.setCorePoolSize(5);
executor.setMaxPoolSize(10);
executor.setQueueCapacity(50);
executor.setThreadNamePrefix("async-");
return executor;
}
6. Web 容器自身线程池
Tomcat/Jetty/Undertow 处理 HTTP 请求的线程池,可在配置文件中调整:
bash
yaml
# Tomcat 配置示例
server:
tomcat:
threads:
max: 200 # 最大工作线程数
min-spare: 10 # 最小空闲线程数
accept-count: 100 # 等待队列长度
7. 数据库连接池
虽然不是线程池,但 HikariCP 等连接池也维护着工作线程,需要合理配置。
bash
yaml
spring:
datasource:
hikari:
maximum-pool-size: 20
minimum-idle: 5
connection-timeout: 30000
最佳实践建议
-
线程池隔离:将 IO 密集型、CPU 密集型、定时任务等不同负载类型的任务分配到不同的线程池
-
命名规范:为每个线程池设置有意义的名称前缀,便于问题排查
-
监控告警:对核心线程池添加监控,关注活跃线程数、队列大小、拒绝任务数等指标
-
拒绝策略:根据业务场景选择合适的拒绝策略:
-
CallerRunsPolicy:降级,由调用线程执行 -
AbortPolicy:抛出异常 -
DiscardPolicy:静默丢弃 -
DiscardOldestPolicy:丢弃最旧的任务
-
-
优雅关闭:在应用关闭时确保线程池正确关闭,避免数据丢失
java
@PreDestroy
public void shutdown() {
businessExecutor().shutdown();
ioExecutor().shutdown();
}
通过合理配置多个线程池,可以让 Spring Boot 应用更好地应对复杂业务场景,提升系统的吞吐量和稳定性。