SpringBoot中ThreadPoolTaskExecutor的使用

简介: SpringBoot中ThreadPoolTaskExecutor的使用

@[TOC]

1 配置自己的线程池

@Configuration
@EnableAsync
public class ThreadPoolConfig
{
   
    // 核心线程池大小
    private int corePoolSize = 50;

    // 最大可创建的线程数
    private int maxPoolSize = 200;

    // 队列最大长度
    private int queueCapacity = 1000;

    // 线程池维护线程所允许的空闲时间
    private int keepAliveSeconds = 300;

    @Bean(name = "threadPoolTaskExecutor")
    public ThreadPoolTaskExecutor threadPoolTaskExecutor()
    {
   
        ThreadPoolTaskExecutor executor = new ThreadPoolTaskExecutor();
        // 设置最大线程池数量
        executor.setMaxPoolSize(maxPoolSize);
        // 核心线程数
        executor.setCorePoolSize(corePoolSize);
        // 线程池维护线程所允许的空闲时间
        executor.setKeepAliveSeconds(keepAliveSeconds);
        // 队列最大长度
        executor.setQueueCapacity(queueCapacity);
        // 线程前缀名称
        executor.setThreadNamePrefix("ThreadPoolTaskExecutor======>");
        // 设置线程池关闭时,等待所有任务都完成后,再继续销毁其他的bean
        // 这样这些异步任务的销毁就会优先于 数据库连接池对象的销毁
        // 也就是说当其他bean都完成任务销毁后,数据库连接池对象才会销毁
        executor.setWaitForTasksToCompleteOnShutdown(true);
        // 任务的等待时间,如果超过这个时间还没有销毁,就强制销毁,以确保应用最后能够被关闭,而不是阻塞住
        executor.setAwaitTerminationSeconds(5);
        // 线程池对拒绝任务(无线程可用)的处理策略
        // 线程不够用时由调用的线程处理该任务
        executor.setRejectedExecutionHandler(new ThreadPoolExecutor.CallerRunsPolicy());
        executor.initialize();
        return executor;
    }
}
  • 一定要自己配置,不要使用SpringBoot自带的,Spring默认的线程池没有上限,并发多时,可能会OOM。并且不是线程重用,每次都会新建一个线程。ThreadPoolTaskExecutor 会重用线程,节省创建线程的资源。因此,在使用ThreadLocal时,也要及时清理缓存,否则,会获取到上一个任务的参数。

    2 使用

    2.1 在Service层使用

  • Controller层直接返回结果,具体的业务,由其他线程执行。
      @Override
      @Async("threadPoolTaskExecutor")
      public void concatAB() {
         
          ......
      }
    

    2.2 多线程中使用事务的写法

  • 事务放到其他方法@Transactional,然后调用。

      @Override
      @Async("threadPoolTaskExecutor")
      public void concatAB() {
         
          addChange()
          ......
      }
    
    @Transactional(rollbackFor = Exception.class,isolation = Isolation.SERIALIZABLE)
      public void addChange(){
         
          .......
      }
    

    2.3 方法内多线程

    2.3.1 错误写法

  • 错误写法

      @Override
      public void concatAB() {
         
          addChange()
          ......
      }
    
      @Async("threadPoolTaskExecutor")
      public void addChange(){
         
          .......
      }
    
  • @Async注解可以让Spring使用线程池来异步执行带有该注解的方法,当一个带有@Async注解的方法被调用时,Spring会为其创建代理对象。通过这个代理对象来调用该方法,Spring就可以控制方法的异步执行了。

  • 如果不是通过代理对象来调用@Async方法,方法的执行将与当前线程同步运行,不会实现异步处理的效果。所以在同一个类中调用加了@Async注解的方法,需要使用代理对象来调用以确保异步生效。

    2.3.2 正确写法(一)

  • 第一种,直接使用线程池执行异步方法。

      @Autowired
      private ThreadPoolTaskExecutor threadPoolTaskExecutor;
    
      @Override
      public void concatAB() {
         
          threadPoolTaskExecutor.execute(() -> {
         
              ....
          });
      }
    

    2.3.2 正确写法(二)

  • 使用代理对象

    @Override
      public void concatAB() {
         
          // 获取代理对象
          IDemoService demoService = (IDemoService) AopContext.currentProxy();
          // 使用代理对象调用
          demoService.addChange()
          ......
      }
    
      @Async("threadPoolTaskExecutor")
      public void addChange(){
         
          .......
      }
    

    2.3.3 正确写法(三)

  • DemoService中通过@Autowired将自身注入到selfProxy属性中,并使用selfProxy对象在syncMethod方法中调用了asyncMethod方法,从而实现了对同一类中的带有@Async注解的方法的异步调用。

    @Autowired
      private DemoService selfProxy;
    
      @Async
      public void asyncMethod() {
         
          // 异步执行的逻辑
      }
    
      public void syncMethod() {
         
          // 调用异步方法需要使用代理对象
          selfProxy.asyncMethod();
      }
    

    3 线程池与并行流的选取

  • IO密集型的业务,选取线程池来完成。(例如,连接数据库之类的操作)
  • cpu密集型的业务,选取并行流allelesvs.parallelStream()的方式处理。(例如,计算类、过滤、求和、分组等)
目录
相关文章
|
Java Spring 容器
Spring boot 自定义ThreadPoolTaskExecutor 线程池并进行异步操作
Spring boot 自定义ThreadPoolTaskExecutor 线程池并进行异步操作
1052 3
|
SQL 消息中间件 JavaScript
性能爆表:SpringBoot利用ThreadPoolTaskExecutor批量插入百万级数据实测!
性能爆表:SpringBoot利用ThreadPoolTaskExecutor批量插入百万级数据实测!
|
1月前
|
JavaScript Java 关系型数据库
基于springboot的项目管理系统
本文探讨项目管理系统在现代企业中的应用与实现,分析其研究背景、意义及现状,阐述基于SSM、Java、MySQL和Vue等技术构建系统的关键方法,展现其在提升管理效率、协同水平与风险管控方面的价值。
|
1月前
|
搜索推荐 JavaScript Java
基于springboot的儿童家长教育能力提升学习系统
本系统聚焦儿童家长教育能力提升,针对家庭教育中理念混乱、时间不足、个性化服务缺失等问题,构建科学、系统、个性化的在线学习平台。融合Spring Boot、Vue等先进技术,整合优质教育资源,提供高效便捷的学习路径,助力家长掌握科学育儿方法,促进儿童全面健康发展,推动家庭和谐与社会进步。
|
1月前
|
JavaScript Java 关系型数据库
基于springboot的古树名木保护管理系统
本研究针对古树保护面临的严峻挑战,构建基于Java、Vue、MySQL与Spring Boot技术的信息化管理系统,实现古树资源的动态监测、数据管理与科学保护,推动生态、文化与经济可持续发展。
|
1月前
|
监控 安全 JavaScript
2025基于springboot的校车预定全流程管理系统
针对传统校车管理效率低、信息不透明等问题,本研究设计并实现了一套校车预定全流程管理系统。系统采用Spring Boot、Java、Vue和MySQL等技术,实现校车信息管理、在线预定、实时监控等功能,提升学校管理效率,保障学生出行安全,推动教育信息化发展。
|
1月前
|
人工智能 Java 关系型数据库
基于springboot的画品交流系统
本项目构建基于Java+Vue+SpringBoot+MySQL的画品交流系统,旨在解决传统艺术交易信息不透明、流通受限等问题,融合区块链与AI技术,实现画品展示、交易、鉴赏与社交一体化,推动艺术数字化转型与文化传播。
|
1月前
|
JavaScript Java 关系型数据库
基于springboot的高校运动会系统
本系统基于Spring Boot、Vue与MySQL,实现高校运动会报名、赛程安排及成绩管理的全流程信息化,提升组织效率,杜绝信息错漏与冒名顶替,推动体育赛事智能化发展。
|
1月前
|
JavaScript 安全 Java
基于springboot的大学生兼职系统
本课题针对大学生兼职信息不对称、权益难保障等问题,研究基于Spring Boot、Vue、MySQL等技术的兼职系统,旨在构建安全、高效、功能完善的平台,提升大学生就业竞争力与兼职质量。