线程池核心原理及使用

java 线程池的核心参数有哪些?各自的作用是什么?

复制代码
public ThreadPoolExecutor(
    int corePoolSize,        // 核心线程数(常驻线程数,即使空闲也不会销毁)
    int maximumPoolSize,     // 最大线程数(线程池允许创建的最大线程数)
    long keepAliveTime,      // 非核心线程空闲超时时间(超时后销毁)
    TimeUnit unit,           // keepAliveTime 的时间单位
    BlockingQueue<Runnable> workQueue, // 任务阻塞队列(核心线程满后,任务入队等待)
    ThreadFactory threadFactory,       // 线程创建工厂(自定义线程命名、优先级等)
    RejectedExecutionHandler handler   // 拒绝策略(队列满+最大线程数满时,处理新任务的策略)
)
  • 阻塞队列:缓冲任务,避免线程创建 / 销毁过于频繁;同时实现「核心线程满 → 入队 → 扩容非核心线程」的流程控制
    • ArrayBlockingQueue(有界数组):推荐生产使用(避免无界队列 OOM);
    • LinkedBlockingQueue(无界链表):Executors 默认使用,易 OOM;
    • SynchronousQueue(同步队列):CachedThreadPool 使用,无存储能力,任务直接交给线程执行
  • 拒绝策略:
    • AbortPolicy(默认):直接抛出 RejectedExecutionException,中断任务提交;
    • CallerRunsPolicy:由提交任务的线程(如主线程)执行任务,降低任务提交速度;
    • DiscardPolicy:静默丢弃新任务,无任何提示;
    • DiscardOldestPolicy:丢弃队列中最旧的任务,尝试提交新任务。(生产中常自定义拒绝策略,如记录日志、异步写入消息队列重试)

执行过程

  1. 当提交任务时,若当前线程数 < 核心线程数(corePoolSize),直接创建核心线程执行任务;

  2. 若当前线程数 ≥ 核心线程数,判断阻塞队列是否已满:

  • 队列未满:将任务放入队列等待执行;

  • 队列已满:判断当前线程数是否 < 最大线程数(maximumPoolSize):

✅ 是:创建非核心线程执行任务;

❌ 否:触发拒绝策略处理任务。

  1. 非核心线程执行完任务后,空闲时间达到 keepAliveTime 则被销毁,线程数回落至 corePoolSize。

线程池类型

FixedThreadPool固定线程池

特点:

  • 线程数固定,任务排队执行,不会创建临时线程;
  • 无界队列可能导致任务堆积,最终 OOM(内存溢出)

适用场景:任务量稳定、需要控制并发数的场景(如服务器接口处理)

java 复制代码
CountDownLatch countDownLatch = new CountDownLatch(7);
ExecutorService executorService = Executors.newFixedThreadPool(7);

CachedThreadPool缓冲线程池

特点:

  • 任务来时直接创建新线程,空闲线程 60 秒销毁;
  • 同步队列无任务堆积,但线程数无上限可能导致创建大量线程,最终 OOM

适用场景:短期、高频、耗时短的任务(如临时批量处理小任务)

java 复制代码
final Semaphore semaphore = new Semaphore(2);
final ExecutorService fixedThread = Executors.newCachedThreadPool();

自定义线程池

java 复制代码
@Configuration
public class ThreadPoolFactory {
    @Bean(name = "openThreadPoll", destroyMethod = "shutdown")
    public ExecutorService openInvoiceThreadPoll() {
        final int corePoolSize = 10;
        final int maximumPoolSize = 10;
        final long keepAliveTime = 0;
        final TimeUnit keepAliveTimeUnit = TimeUnit.SECONDS;
        return new ThreadPoolExecutor(corePoolSize, maximumPoolSize, keepAliveTime, keepAliveTimeUnit,
                new LinkedBlockingQueue<Runnable>(), new NamedThreadFactory("openInvoiceThreadPoll"),
                new ThreadPoolExecutor.CallerRunsPolicy()) ;
    }
 }

@Autowired
private ExecutorService openThreadPoll;

openThreadPoll.submit(() -> {
	logiInterface(entity,"aaa","bbb,"ccc,"0");
});

其他

并发和并行的区别?

  • 并发:单核CPU同一时间段内交替执行多个任务
  • 并发:多核CPU同一时间同时执行多个任务
相关推荐
侠客行03175 小时前
Mybatis连接池实现及池化模式
java·mybatis·源码阅读
蛇皮划水怪5 小时前
深入浅出LangChain4J
java·langchain·llm
灰子学技术6 小时前
go response.Body.close()导致连接异常处理
开发语言·后端·golang
老毛肚6 小时前
MyBatis体系结构与工作原理 上篇
java·mybatis
风流倜傥唐伯虎7 小时前
Spring Boot Jar包生产级启停脚本
java·运维·spring boot
二十雨辰7 小时前
[python]-AI大模型
开发语言·人工智能·python
Yvonne爱编码7 小时前
JAVA数据结构 DAY6-栈和队列
java·开发语言·数据结构·python
Re.不晚7 小时前
JAVA进阶之路——无奖问答挑战1
java·开发语言
你这个代码我看不懂7 小时前
@ConditionalOnProperty不直接使用松绑定规则
java·开发语言
pas1367 小时前
41-parse的实现原理&有限状态机
开发语言·前端·javascript