线程池核心原理及使用

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同一时间同时执行多个任务
相关推荐
郝学胜-神的一滴1 小时前
Effective Modern C++ 条款34:优先考虑lambda而非std::bind
开发语言·c++·程序人生
路边草随风1 小时前
java 实现 flink 读 kafka 写 delta
java·大数据·flink·kafka
逆风局?1 小时前
后端Web实战(部门管理)——日志技术
java·前端
小马爱打代码1 小时前
Spring AI:ChatClient实现对话效果
java·人工智能·spring
无敌最俊朗@1 小时前
C++ 内存管理与编译原理 (面试复习2)
java·开发语言·jvm
火山灿火山1 小时前
Qt信号和槽
开发语言·qt
赴前尘1 小时前
docker 配置ipv6地址
java·docker·容器
开开心心就好1 小时前
图片批量压缩工具:支持有损无损两种模式
java·游戏·pdf·excel·散列表·启发式算法·1024程序员节
Overt0p1 小时前
博客系统(2)
java