CompletableFuture task2 = asyncTasks.doTaskOne(“2”);
CompletableFuture task3 = asyncTasks.doTaskOne(“3”);
CompletableFuture.allOf(task1, task2, task3).join();
return “”;
}
@GetMapping(“/api-2”)
public String taskTwo() {
CompletableFuture task1 = asyncTasks.doTaskTwo(“1”);
CompletableFuture task2 = asyncTasks.doTaskTwo(“2”);
CompletableFuture task3 = asyncTasks.doTaskTwo(“3”);
CompletableFuture.allOf(task1, task2, task3).join();
return “”;
}
}
上面的代码中,有两个API接口,这两个接口的具体执行逻辑中都会把执行过程拆分为三个异步任务来实现。
好了,思考一分钟,想一下。如果这样实现,会有什么问题吗?
上面这段代码,在API请求并发不高,同时如果每个任务的处理速度也够快的时候,是没有问题的。但如果并发上来或其中某几个处理过程扯后腿了的时候。这两个提供不相干服务的接口可能会互相影响。比如:假设当前线程池配置的最大线程数有2个,这个时候/api-1接口中task1和task2处理速度很慢,阻塞了;那么此时,当用户调用api-2接口的时候,这个服务也会阻塞!
造成这种现场的原因是:默认情况下,所有用@Async
创建的异步任务都是共用的一个线程池,所以当有一些异步任务碰到性能问题的时候,是会直接影响其他异步任务的。
为了解决这个问题,我们就需要对异步任务做一定的线程池隔离,让不同的异步任务互不影响。
下面,我们就来实际操作一下!
第一步:初始化多个线程池,比如下面这样:
@EnableAsync
@Configuration
public class TaskPoolConfig {
@Bean
public Executor taskExecutor1() {
ThreadPoolTaskExecutor executor = new ThreadPoolTaskExecutor();
executor.setCorePoolSize(2);
executor.setMaxPoolSize(2);
executor.setQueueCapacity(10);
executor.setKeepAliveSeconds(60);
executor.setThreadNamePrefix(“executor-1-”);
executor.setRejectedExecutionHandler(new ThreadPoolExecutor.CallerRunsPolicy());
return executor;
}
@Bean
public Executor taskExecutor2() {
ThreadPoolTaskExecutor executor = new ThreadPoolTaskExecutor();
executor.setCorePoolSize(2);
executor.setMaxPoolSize(2);
executor.setQueueCapacity(10);
executor.setKeepAliveSeconds(60);
executor.setThreadNamePrefix(“executor-2-”);
executor.setRejectedExecutionHandler(new ThreadPoolExecutor.CallerRunsPolicy());
return executor;
}
}
注意:这里特地用
executor.setThreadNamePrefix
设置了线程名的前缀,这样可以方便观察后面具体执行的顺序。
第二步:创建异步任务,并指定要使用的线程池名称
@Slf4j
@Component
public class AsyncTasks {
public static Random random = new Random();
@Async(“taskExecutor1”)
public CompletableFuture doTaskOne(String taskNo) throws Exception {
log.info(“开始任务:{}”, taskNo);
long start = System.currentTimeMillis();
Thread.sleep(random.nextInt(10000));
long end = System.currentTimeMillis();
log.info(“完成任务:{},耗时:{} 毫秒”, taskNo, end - start);
return CompletableFuture.completedFuture(“任务完成”);
}
@Async(“taskExecutor2”)
public CompletableFuture doTaskTwo(String taskNo) throws Exception {
log.info(“开始任务:{}”, taskNo);
long start = System.currentTimeMillis();
Thread.sleep(random.nextInt(10000));
long end = System.currentTimeMillis();
log.info(“完成任务:{},耗时:{} 毫秒”, taskNo, end - start);
return CompletableFuture.completedFuture(“任务完成”);
}
}
这里@Async
注解中定义的taskExecutor1
和taskExecutor2
就是线程池的名字。由于在第一步中,我们没有具体写两个线程池Bean的名称,所以默认会使用方法名,也就是taskExecutor1
和taskExecutor2
。
第三步:写个单元测试来验证下,比如下面这样:
@Slf4j
@SpringBootTest
public class Chapter77ApplicationTests {
@Autowired
private AsyncTasks asyncTasks;
@Test
public void test() throws Exception {
long start = System.currentTimeMillis();
// 线程池1
CompletableFuture task1 = asyncTasks.doTaskOne(“1”);
CompletableFuture task2 = asyncTasks.doTaskOne(“2”);
CompletableFuture task3 = asyncTasks.doTaskOne(“3”);
// 线程池2
CompletableFuture task4 = asyncTasks.doTaskTwo(“4”);
CompletableFuture task5 = asyncTasks.doTaskTwo(“5”);
CompletableFuture task6 = asyncTasks.doTaskTwo(“6”);
// 一起执行
CompletableFuture.allOf(task1, task2, task3, task4, task5, task6).join();
自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。
深知大多数Java工程师,想要提升技能,往往是自己摸索成长或者是报班学习,但对于培训机构动则几千的学费,着实压力不小。自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!
因此收集整理了一份《2024年Java开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。
既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Java开发知识点,真正体系化!
由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!
如果你觉得这些内容对你有帮助,可以扫码获取!!(备注Java获取)
![img](https://img-blog.csdnimg.cn/img_convert/b239abf32f273d646fc31f15176097d1.jpeg)
最后
笔者已经把面试题和答案整理成了面试专题文档
《互联网大厂面试真题解析、进阶开发核心学习笔记、全套讲解视频、实战项目源码讲义》点击传送门即可获取!
3435)]
[外链图片转存中…(img-8sLMMBBw-1713390183435)]
[外链图片转存中…(img-Ft4qLGHU-1713390183435)]
[外链图片转存中…(img-k5aPoJQZ-1713390183436)]
[外链图片转存中…(img-mYzavRmq-1713390183436)]
《互联网大厂面试真题解析、进阶开发核心学习笔记、全套讲解视频、实战项目源码讲义》点击传送门即可获取!