线程池个人笔记总结

线程池的创建示例
复制代码
@Override
    @Bean(name = "threadPool")
    public ThreadPoolTaskExecutor getAsyncExecutor() {
        ThreadPoolTaskExecutor threadPool = new ThreadPoolTaskExecutor();
        // 设置核心线程数
        threadPool.setCorePoolSize(10);
        // 设置最大线程数
        threadPool.setMaxPoolSize(20);
        // 线程池所使用的缓冲队列
        threadPool.setQueueCapacity(10000);
        // 等待任务在关机时完成--表明等待所有线程执行完
        threadPool.setWaitForTasksToCompleteOnShutdown(true);
        // 线程池关闭的等待时间(默认为0)
        threadPool.setAwaitTerminationSeconds(60*5);
        // 非核心线程空闲的存活时间
        threadPool.setKeepAliveSeconds(1);
        // 线程名称前缀
        threadPool.setThreadNamePrefix("newton-thread-");
        // 线程池拒绝策略
        threadPool.setRejectedExecutionHandler(new ThreadPoolExecutor.CallerRunsPolicy());
        // 初始化线程
        threadPool.initialize();
        return threadPool;
    }
线程池参数解释
KeepAliveTime是什么,有什么用

我们设置的KeepAliveTime的时间是非核心线程空闲的存活时间,如果超过存活时间则会被销毁目的为了节省系统资源

拒绝策略的触发条件

当任务队列已满并且线程池中的线程数达到最大线程数时,新提交的任务就会触发拒绝策略

任务队列什么时候有数据

只有当线程池的使用线程数达到最大线程数时(包括核心和非核心线程数),此时新的任务才会进入任务队列等待执行

核心线程和非核心线程有什么区别

首先他们运行速度没有区别,行速度取决于CPU的调度和处理能力,以及任务本身的复杂性和计算量,他们区别主要在核心线程不会被销毁,除非线程池关闭,非核心线程是达到一定条件额外创建的,并且超过设置的存活时间会被销毁来回收系统资源。

线程池执行流程
  • 先判断核心线程是否满了,如果没满就创建核心线程执行任务
  • 如果核心线程满了,判断最大线程是否满了,如果没满,就创建非核心线程执行任务
  • 判断非核心线程是否超过存活时间,如果超过,就销毁这个线程
  • 如果线程池中的使用的线程数达到最大线程数,新的任务就会进入等待队列
  • 如果等待队列满了和线程池都满了,则调用拒绝策略
线程池拒绝策略
AbortPolicy(默认策略):

抛出RejectedExecutionException异常,表示拒绝执行新任务。

CallerRunsPolicy:

将任务返回给调用线程执行,如果线程池已关闭,则丢弃任务。

DiscardPolicy:

默默丢弃无法处理的任务,不给予任何处理。

DiscardOldestPolicy:

丢弃任务队列中等待时间最久的任务,然后将新任务加入队列。

自定义拒绝策略

一般需要实现RejectedExecutionHandler接口。以下是一个自定义的拒绝策略示例:

复制代码
import java.util.concurrent.RejectedExecutionHandler;
import java.util.concurrent.ThreadPoolExecutor;

public class CustomRejectedExecutionHandler implements RejectedExecutionHandler {

    @Override
    public void rejectedExecution(Runnable r, ThreadPoolExecutor executor) {
        // 自定义拒绝策略,例如记录日志或将任务加入队列等
        System.err.println("Task rejected. " + r.toString());
    }
}

然后可以在创建线程池时使用自定义拒绝策略:

复制代码
ThreadPoolExecutor executor = new ThreadPoolExecutor(
    corePoolSize, maxPoolSize, keepAliveTime, TimeUnit.SECONDS,
    new ArrayBlockingQueue<>(queueCapacity), new CustomRejectedExecutionHandler());
线程池常用方法
shutdown

shutdown() 方法会优雅地关闭线程池,等待已提交的任务执行完成(可以设置最长等待时间),不再接受新任务。

shutdownNow

shutdownNow() 会先改状态为STOP不接收新的任务,并尝试取消正在执行的任务,立即关闭线程池 并返回未执行的任务列表

CountDownLatch

CountDownLatch是一种倒计时锁,允许一个或多个线程等待其他线程执行完毕才运行

复制代码
// 创建countDownLatch 数量为线程数
CountDownLatch countDownLatch = new CountDownLatch(10);
for (int i = 1; i <= maxPage; i++) {
     executor.execute(() -> {
         try{
             // 开线程执行业务逻辑
             xxx
         }catch{
             
         } finally {
             // 执行完一个线程任务就在finally中进行减1
    		countDownLatch.countDown();
    	}
    });
}
// 等待所有子线程执行完毕前会一直阻塞
countDownLatch.await();
// 后续主线程任务
xxx
setDefaultUncaughtExceptionHandler

用于为所有新创建的线程设置默认的未捕获异常处理器

相关推荐
Leo.yuan3 分钟前
数据分析怎么做?高效的数据分析方法有哪些?
大数据·数据库·信息可视化·数据挖掘·数据分析
强化学习与机器人控制仿真5 分钟前
openpi 入门教程
开发语言·人工智能·python·深度学习·神经网络·机器人·自动驾驶
zm24 分钟前
网络编程epoll和udp
服务器·网络·数据库
野犬寒鸦27 分钟前
Linux常用命令详解(下):打包压缩、文本编辑与查找命令
linux·运维·服务器·数据库·后端·github
明月看潮生41 分钟前
青少年编程与数学 02-019 Rust 编程基础 08课题、字面量、运算符和表达式
开发语言·青少年编程·rust·编程与数学
什码情况1 小时前
星际篮球争霸赛/MVP争夺战 - 华为OD机试真题(A卷、Java题解)
java·数据结构·算法·华为od·面试·机试
天天打码1 小时前
Rspack:字节跳动自研 Web 构建工具-基于 Rust打造高性能前端工具链
开发语言·前端·javascript·rust·开源
Petrichorzncu1 小时前
Lua再学习
开发语言·学习·lua
AA-代码批发V哥1 小时前
正则表达式: 从基础到进阶的语法指南
java·开发语言·javascript·python·正则表达式
字节高级特工1 小时前
【C++】”如虎添翼“:模板初阶
java·c语言·前端·javascript·c++·学习·算法