在Springboot中使用ThreadPoolExecutor线程池

在springboot项目中如果需要用到ThreadPoolExecutor线程池的话是非常方便的。比使用java并发包中的Executors都还方便很多。
实际上spring中的线程池ThreadpoolExecutor只是对java并发包中的线程池的封装。这样便于在spring环境中快速使用。通过几个注解即可,降低了对代码的侵入性。

1.ThreadPoolExecutor配置

在本文中,将使用spring提供的ThreadPoolTaskExecutor进行配置。

1.1 yml中的配置参数

配置参数如下:

#线程池配置
async.executor.thread.core_pool_size: 5
# 最大线程数
async.executor.thread.max_pool_size: 20
# 任务队列大小
async.executor.thread.queue_capacity: 100
# 线程池中线程的名称前缀
async.executor.thread.name.prefix: async-service-
# 缓冲队列中线程的空闲时间
async.executor.thread.keep_alive_seconds: 100

1.2 Executor的Bean

在Spring中通过@Configuration配置一个自定义的Bean,来启动Executor。

package com.dhb.gts.javacourse.week7;


import lombok.extern.slf4j.Slf4j;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.core.task.TaskExecutor;
import org.springframework.scheduling.annotation.EnableAsync;
import org.springframework.scheduling.concurrent.ThreadPoolTaskExecutor;

import java.util.concurrent.ThreadPoolExecutor;

@Configuration
@EnableAsync
@Slf4j
public class BeanConfig {

	@Value("${async.executor.thread.core_pool_size}")
	private int corePoolSize;
	@Value("${async.executor.thread.max_pool_size}")
	private int maxPoolSize;
	@Value("${async.executor.thread.queue_capacity}")
	private int queueCapacity;
	@Value("${async.executor.thread.name.prefix}")
	private String namePrefix;
	@Value("${async.executor.thread.keep_alive_seconds}")
	private int keepAliveSeconds;

	@Bean
	public TaskExecutor taskExecutor() {
		ThreadPoolTaskExecutor executor = new ThreadPoolTaskExecutor();
		// 设置核心线程数
		executor.setCorePoolSize(corePoolSize);
		// 设置最大线程数
		executor.setMaxPoolSize(maxPoolSize);
		// 设置队列容量
		executor.setQueueCapacity(queueCapacity);
		// 设置线程活跃时间(秒)
		executor.setKeepAliveSeconds(keepAliveSeconds);
		// 设置默认线程名称
		executor.setThreadNamePrefix(namePrefix);
		// 设置拒绝策略
		executor.setRejectedExecutionHandler(new ThreadPoolExecutor.CallerRunsPolicy());
		// 等待所有任务结束后再关闭线程池
		executor.setWaitForTasksToCompleteOnShutdown(true);
		log.info("创建一个线程池 corePoolSize is [" + corePoolSize + "] maxPoolSize is [" + maxPoolSize + "] queueCapacity is [" + queueCapacity +
				"] keepAliveSeconds is [" + keepAliveSeconds + "] namePrefix is [" + namePrefix + "].");
		return executor;
	}
}

上述代码中用到的注解:
@Configuration:Spring 容器在启动时,会加载带有 @Configuration 注解的类,对其中带有 @Bean 注解的方法进行处理,实例化一个Bean。
@Bean:是一个方法级别上的注解,用以在代码中配置一个Bean,主要用在 @Configuration 注解的类里,也可以用在 @Component 注解的类里。添加的 bean 的 id 为方法名。
@Value:从yaml或者配置文件中去寻找指定标识的属性值。

2.线程的调用方式

使用线程持的时候,通过@Async进行配置,如果@Async不指定执行的线程池name,这个注解配置的方法将会自动提交到一个name为taskExecutor的线程池去执行。
如果taskExecutor不存在,则会去寻找ThreadPoolTaskExecutor类型的对象执行。
线程的调用方式有两种,分别是同步调用和异步调用。本文通过springboot的web项目中来演示。

2.1 异步调用

在OrderService中的方法为:

	@Async
	public void asyncInsertRandomOrder(){
		Stopwatch stopwatch = Stopwatch.createStarted();
		log.info("开始通过线程持进行异步插入!");
		int orderNo = getMaxOderNo();
		insertOrder(orderNo+1);
		log.info("通过线程池插入完成,共耗时:"+stopwatch.stop());
	}

异步调用的方法没有返回值。
现在在OrderController中使用:

	@RequestMapping("/asyncInsertRandomOrder")
	public String asyncInsertRandomOrder() {
		orderService.asyncInsertRandomOrder();
		log.info("调用异步方法插入一个订单!");
		return "success";
	}

只需要将OrderService注入到OrderController即可使用。

2.2 同步调用

在OrderService中的方法为:

	@Async
	public ListenableFuture<OrderSummaryEntity> asyncQueryOrderById(int order_id){
		OrderSummaryEntity entity = orderSummaryDao.selectById(order_id);
		return new AsyncResult<>(entity);
	}

需要注意的是,同步调用方法需要得到返回值,将通过ListenableFuture类返回。返回的结果是一个AsyncResult对象。
在OrderController中使用:

	@RequestMapping("/asyncQueryByKey")
	public String asyncQueryByKey(String key) {
		Stopwatch stopwatch = Stopwatch.createStarted();
		Integer orde_id = Integer.parseInt(key);
		OrderSummaryEntity entity = null;
		try {
			entity = orderService.asyncQueryOrderById(orde_id).get();
		}catch (Exception e) {
			e.printStackTrace();
		}
		stopwatch.stop();
		log.info("通过key查询,走索引耗时:" + stopwatch);
		return JSON.toJSONString(entity);
	}

此处需要用get()方法,之后就会被阻塞,等待线程持中的方法执行完成。

  • 0
    点赞
  • 10
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
Spring Boot创建线程池有多种方式,最常用的是通过配置文件或者Java配置类创建ThreadPoolTaskExecutor来实现。以下是两种常用的方式: 1. 通过配置文件创建线程池 在application.properties或application.yml文件添加以下配置: ``` # 线程池配置 spring.task.execution.pool.core-size=10 spring.task.execution.pool.max-size=20 spring.task.execution.pool.queue-capacity=200 ``` 这里配置了线程池的核心线程数、最大线程数和队列容量。然后在代码注入ThreadPoolTaskExecutor,即可使用线程池了。 ```java @Autowired private ThreadPoolTaskExecutor taskExecutor; public void doSomething() { taskExecutor.execute(new Runnable() { @Override public void run() { // 执行任务 } }); } ``` 2. 通过Java配置类创建线程池 通过创建一个Java配置类来配置线程池。 ```java @Configuration @EnableAsync public class TaskExecutePool { @Bean("taskExecutor") public Executor taskExecutor() { ThreadPoolTaskExecutor executor = new ThreadPoolTaskExecutor(); executor.setCorePoolSize(10); executor.setMaxPoolSize(20); executor.setQueueCapacity(200); executor.setThreadNamePrefix("taskExecutor-"); executor.setRejectedExecutionHandler(new ThreadPoolExecutor.CallerRunsPolicy()); executor.initialize(); return executor; } } ``` 这里通过@Bean注解将ThreadPoolTaskExecutor注入到Spring容器,然后在需要使用线程池的地方通过@Resource或@Autowired注解注入即可。 ```java @Resource(name="taskExecutor") private ThreadPoolTaskExecutor taskExecutor; public void doSomething() { taskExecutor.execute(new Runnable() { @Override public void run() { // 执行任务 } }); } ``` 这样就可以在Spring Boot创建一个线程池了。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值