线程池核心原理及使用

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 小时前
Java 性能优化实战:如何将海量扁平数据高效转化为类目字典树?
java·开发语言·json
SmartRadio1 小时前
ESP32-S3 双模式切换实现:兼顾手机_路由器连接与WiFi长距离通信
开发语言·网络·智能手机·esp32·长距离wifi
laowangpython1 小时前
Rust 入门:GitHub 热门内存安全编程语言
开发语言·其他·rust·github
我叫汪枫1 小时前
在后台管理系统中,如何递归和选择保留的思路来过滤菜单
开发语言·javascript·node.js·ecmascript
_.Switch1 小时前
东方财富股票数据JS逆向:secids字段和AES加密实战
开发语言·前端·javascript·网络·爬虫·python·ecmascript
软件技术NINI1 小时前
webkit简介及工作流程
开发语言·前端·javascript·udp·ecmascript·webkit·yarn
Brendan_0011 小时前
JavaScript的Stomp.over
开发语言·javascript·ecmascript
念2341 小时前
f5 shape分析
开发语言·javascript·ecmascript
苍穹之跃1 小时前
某量JS逆向
开发语言·javascript·ecmascript
思茂信息1 小时前
CST软件如何进行参数化扫描?
运维·开发语言·javascript·windows·ecmascript·软件工程·软件需求