SpringBoot线程池使⽤详解
前提摘要:
基于Springboot 2.1.4.RELEASE
▎ 配置TaskExecutor
import urrent.ThreadPoolExecutor;
import t.annotation.Bean;
import t.annotation.Configuration;
import ask.TaskExecutor;
import org.springframework.scheduling.annotation.EnableAsync;
import org.urrent.ThreadPoolTaskExecutor;
@Configuration
@EnableAsync
public class ThreadPoolConfig {
@Bean
public TaskExecutor taskExecutor() {
ThreadPoolTaskExecutor executor = new ThreadPoolTaskExecutor();
// 设置核⼼线程数
executor.setCorePoolSize(5);
// 设置最⼤线程数
executor.setMaxPoolSize(10);
// 设置队列容量
executor.setQueueCapacity(5);
/
/ 设置线程活跃时间,单位秒
executor.setKeepAliveSeconds(60);
// 设置核⼼线程超时回收
executor.setAllowCoreThreadTimeOut(true);
// 设置默认线程名称
executor.setThreadNamePrefix("IThread-");
// 设置拒绝策略
executor.setRejectedExecutionHandler(new ThreadPoolExecutor.DiscardPolicy());
return executor;
}
}
▎ 参数解读
● CorePoolSize
核⼼线程数,核⼼线程会⼀直存活,即使没有任务需要处理。当线程数⼩于核⼼线程数时,即使现有的线程空闲,线程池也会优先创建新线程来处理任务,⽽不是直接交给现有的线程处理。
核⼼线程在allowCoreThreadTimeout被设置为true时会超时退出,默认情况下不会退出。
● MaxPoolSize
当线程数⼤于或等于核⼼线程,且任务队列已满时,线程池会创建新的线程,直到线程数量达到maxPoolSize。如果线程数已等于maxPoolSize,且任务队列已满,则已超出线程池的处理能⼒,线程池会拒绝处理任务⽽抛出异常。
● queueCapacity
任务队列容量。从maxPoolSize的描述上可以看出,任务队列的容量会影响到线程的变化,因此任务队列的长度也需要恰当的设置。
● keepAliveTime
当线程空闲时间达到keepAliveTime,该线程会退出,直到线程数量等于corePoolSize。如果allowCoreThreadTimeout设置为true,则所有线程均会退出直到线程数量为0。
● allowCoreThreadTimeout
是否允许核⼼线程空闲退出,默认值为false。
● RejectedExecutionHandler
拒绝策略:当线程数⼤于MaxPoolSize+queueCapacity被触发:
☞ CallerRunsPolicy - 当触发拒绝策略,只要线程池没有关闭的话,则使⽤调⽤线程直接运⾏任务。⼀般并发⽐较⼩,性能要求不⾼,不允许失败。但是,由于调⽤者⾃⼰运⾏任务,如果任务提交速度过快,可能导致程序阻塞,性能效率上必然的损失较⼤
☞ AbortPolicy - 丢弃任务,并抛出拒绝执⾏ RejectedExecutionException 异常信息。线程池默认的拒绝策略。必须处理好抛出的异常,否则会打断当前的执⾏流程,影响后续的任务执⾏。
☞ DiscardPolicy - 直接丢弃,其他啥都没有
☞ DiscardOldestPolicy - 当触发拒绝策略,只要线程池没有关闭的话,丢弃阻塞队列 workQueue 中最⽼的⼀个任务,并将新任务加⼊
▎ 线程池执⾏流程图 图⽚来源:
▎ 案例: 使⽤AbortPolicy拒绝策略,模拟⾼并发触发异常
☞ TaskExecutor配置
@Bean
public TaskExecutor taskExecutor() {
ThreadPoolTaskExecutor executor = new ThreadPoolTaskExecutor();
// 设置核⼼线程数
executor.setCorePoolSize(5);
// 设置最⼤线程数
executor.setMaxPoolSize(10);
// 设置队列容量
executor.setQueueCapacity(5);
/
/ 设置线程活跃时间,单位秒
executor.setKeepAliveSeconds(60);
// 设置核⼼线程超时回收
executor.setAllowCoreThreadTimeOut(true);
// 设置默认线程名称
executor.setThreadNamePrefix("IThread-");
// 设置拒绝策略
executor.setRejectedExecutionHandler(new ThreadPoolExecutor.AbortPolicy()); return executor;
}
☞ 创建异步任务
ExecutorService
public interface ExecutorService {
public void exec();
}
ExecutorServiceImpl
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.scheduling.annotation.Async;
import org.springframework.stereotype.Service;
service.ExecutorService;
@Service("executorService")
public class ExecutorServiceImpl implements ExecutorService {
private Logger log = Logger(getClass());
@Override
@Async
public void exec() {
log.info(Thread.currentThread().getName() + "开始执⾏");
try {
// 模拟业务处理耗时
Thread.sleep(10000);
} catch (InterruptedException e) {
e.printStackTrace();
}
log.info("<<<<<<<<<<<<<<<<;线程执⾏完毕>>>>>>>>>>>>>>>>");
}
}
☞ 编写Controller,调⽤异步任务
ExecutorController
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RestController;
service.ExecutorService;
@RestController
public class ExecutorController {
@Autowired
private ExecutorService executorService;
@GetMapping("/executor")springboot是啥
public String executor() {
try {
<();
return "success";
} catch (Exception e) {
e.printStackTrace();
}
return "error";
}
}
☞ 打开浏览器,访问Controller接⼝,不断刷新模拟⾼并发,观察返回结果,如果出现error说明策略被触发了
版权声明:本站内容均来自互联网,仅供演示用,请勿用于商业和其他非法用途。如果侵犯了您的权益请与我们联系QQ:729038198,我们将在24小时内删除。
发表评论