SpringBoot自定义线程池

本文介绍了如何在SpringBoot应用中使用线程处理异步任务,包括使用@Async注解启用异步,以及自定义线程池(通过application.yml配置或自定义AsyncConfigurer接口)以避免性能问题。
摘要由CSDN通过智能技术生成

使用SpringBoot构建应用时,如何使用线程处理异步任务?其实springBoot已经提供了默认的实现,通过在启动类上加上注解@EnableAsync, 然后在需要异步处理的方法上增加注解@Async即可启动一个线程进行异步处理。其实质类似于:new Thread(()-{System.out.print(“处理异步任务”)}).start()。但需要注意的是@Async默认使用的是SimpleAsyncTaskExecutor,每次提交任务都创建线程,要小心在使用大任务的场景下创建大量线程导致OOM异常。所以,还是自定义线程池好。自定义线程池有2种实现方式。

1. 自定义线程池,然后使用自己定义的线程池

1.1 修改application.yml文件,增加线程池的配置参数,如下

在这里插入图片描述

1.2 线程池配置属性类MyThreadPoolConfig .java

/**
* 线程池配置属性类
*/

@ConfigurationProperties(prefix = "mytask.pool")

public class MyThreadPoolConfig {
	private int corePoolSize;
	
	private int maxPoolSize;
	
	private int keepAliveSeconds;
	
	private int queueCapacity;

}

1.3 启动类上一定要开启线程异步支持:@EnableAsync

@EnableAsync
@EnableConfigurationProperties({MyThreadPoolConfig.class})
@SpringCloudApplication

public class application{
	public static void main(String[] args){
		SpringApplication.run(application.class, args);
	}
}

1.4 创建自己的线程池

/**
 * 创建线程池
 */
@Configuration
public class MyTaskExecutePool {
 
    @Autowired
    private MyThreadPoolConfig config;
 
    @Bean("myTaskPool")
    public Executor taskExecutor() {
        ThreadPoolTaskExecutor executor = new ThreadPoolTaskExecutor();
        //核心线程池大小
        executor.setCorePoolSize(config.getCorePoolSize());
        //最大线程数
        executor.setMaxPoolSize(config.getMaxPoolSize());
        //队列容量
        executor.setQueueCapacity(config.getQueueCapacity());
        //活跃时间
        executor.setKeepAliveSeconds(config.getKeepAliveSeconds());
        //线程名字前缀
        executor.setThreadNamePrefix("TaskExecutePool-");
 
        // setRejectedExecutionHandler:当pool已经达到maxSize的时候,如何处理新进任务
        // CallerRunsPolicy:不在新线程中执行任务,而是由调用者所在的线程来执行
        executor.setRejectedExecutionHandler(new ThreadPoolExecutor.CallerRunsPolicy());
         // 等待所有任务结束后再关闭线程池
        executor.setWaitForTasksToCompleteOnShutdown(true);
        executor.initialize();
        return executor;
    }
}

需要注意的是这样定义的线程池在使用的时候要在@Async主键里指定名称,如:@Async(“mytaskExecutor”), 否则会使用spingtBoot提供的默认线程池。

2. 重新spring默认的线程池

第一种方式创建的线程池在使用的时候必须要指明异步任务要使用的线程池名称,而重新srping默认的线程池,则不需要指定名称,直接写@Asyncj即可。

配置方式和方式一类似,唯一的区别在于配置类要实现AsyncConfigurer

/**
* 重新SpringBoot默认的线程池
*/
@Configuration
public class OverrideAsyncTaskExecutePool implements AsyncConfigurer{


//注入配置类
@Autowired
MyThreadPoolConfig config;

@Override
public Executor getAsyncExecutor() {
	ThreadPoolTaskExecutor executor = new ThreadPoolTaskExecutor();
	//核心线程池大小
	executor.setCorePoolSize(config.getCorePoolSize());
	//最大线程数
	executor.setMaxPoolSize(config.getMaxPoolSize());
	//队列容量
	executor.setQueueCapacity(config.getQueueCapacity());
	//活跃时间
	executor.setKeepAliveSeconds(config.getKeepAliveSeconds());
	//线程名字前缀
	executor.setThreadNamePrefix("MyExecutor-");
	
	// setRejectedExecutionHandler:当pool已经达到max size的时候,如何处理新任务
	// CallerRunsPolicy:不在新线程中执行任务,而是由调用者所在的线程来执行
	executor.setRejectedExecutionHandler(new ThreadPoolExecutor.CallerRunsPolicy());
	executor.initialize();
	return executor;
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值