CompletableFuture异步编排在多任务并行执行中的使用

将需要串行执行的任务,改成并行执行,减少接口调用的响应时间,提高接口性能

简单举例:

查询标签列表需要调用不同系统的不同服务接口来获取,串行调用会很耗费时间

改成下图调用方式:

串行调用改成并行调用,减少了服务间相互调用的等待时间,提升了接口性能

代码示例:
/**
 * 自定义线程池配置
 */
@Configuration
public class ThreadPoolConfig {
 
    // 核心线程数 20
    @Value("${lease.thread.core-size}")
    private Integer coreSize;
 
    // 最大线程数 100
    @Value("${lease.thread.max-size}")
    private Integer maxSize;
 
    // 空闲线程存活时间 10
    @Value("${lease.thread.keep-alive-time}")
    private Integer keepAliveTime;
 
    @Bean
    public ThreadPoolExecutor threadPoolExecutor() {
        return new ThreadPoolExecutor(coreSize,
                maxSize,
                keepAliveTime,
                TimeUnit.SECONDS,
                new LinkedBlockingQueue<>(100000),
                Executors.defaultThreadFactory(),
                new ThreadPoolExecutor.AbortPolicy());
    }
}
使用CompletableFuture和线程池,改写查询方法
@Service
public class RoomInfoServiceImpl extends ServiceImpl<RoomInfoMapper, RoomInfo>
        implements RoomInfoService {
 
    @Autowired
    private RoomInfoMapper roomInfoMapper;
 
    ......
 
    @Autowired
    private RoomLeaseTermService roomLeaseTermService;
 
    @Autowired
    private ThreadPoolExecutor threadPoolExecutor;
 
 
    @Override
    public RoomDetailVo getDetailById(Long id) {
        RoomDetailVo resultVo = new RoomDetailVo();
        // 获取房间详情和公寓信息
        CompletableFuture<Void> completableFuture = CompletableFuture.supplyAsync(() -> {
            RoomInfo roomInfo = this.getById(id);
            BeanUtils.copyProperties(roomInfo, resultVo);
            return roomInfo;
        }, threadPoolExecutor).thenAcceptAsync((roomInfo) -> {
            resultVo.setApartmentInfo(apartmentInfoService.getById(roomInfo.getApartmentId()));
        }, threadPoolExecutor);
        // 获取图片列表
        CompletableFuture<Void> graphVosFuture = CompletableFuture.runAsync(() -> {
            List<GraphInfo> graphInfos = graphInfoService.lambdaQuery()
                    .eq(GraphInfo::getItemId, id)
                    .eq(GraphInfo::getItemType, ItemType.ROOM)
                    .list();
            List<GraphVo> graphVos = graphInfos.stream().map(item -> {
                GraphVo graphVo = new GraphVo();
                BeanUtils.copyProperties(item, graphVo);
                return graphVo;
            }).collect(Collectors.toList());
            resultVo.setGraphVoList(graphVos);
        }, threadPoolExecutor);
        // 获取属性信息列表
        CompletableFuture<Void> attrValueVosFuture = CompletableFuture.runAsync(() -> {
            resultVo.setAttrValueVoList(attrValueMapper.getAttrValueVosByRoomId(id));
        }, threadPoolExecutor);
        // 获取配套信息列表
        CompletableFuture<Void> facilityInfoFuture = CompletableFuture.runAsync(() -> {
            resultVo.setFacilityInfoList(facilityInfoMapper.getFacilityInfos(id));
        }, threadPoolExecutor);
        // 获取标签信息列表
        CompletableFuture<Void> labelInfoFuture = CompletableFuture.runAsync(() -> {
            resultVo.setLabelInfoList(labelInfoMapper.getLabelInfos(id));
        }, threadPoolExecutor);
        // 获取支付方式列表
        CompletableFuture<Void> paymentTypeFuture = CompletableFuture.runAsync(() -> {
            resultVo.setPaymentTypeList(paymentTypeMapper.getPaymentTypes(id));
        }, threadPoolExecutor);
        // 获取可选租期列表
        CompletableFuture<Void> leaseTermFuture = CompletableFuture.runAsync(() -> {
            resultVo.setLeaseTermList(leaseTermMapper.getLeaseTerms(id));
        }, threadPoolExecutor);
        // 等待所有任务全部完成
        try {
            CompletableFuture.allOf(completableFuture,
                    graphVosFuture,
                    attrValueVosFuture,
                    facilityInfoFuture,
                    labelInfoFuture,
                    paymentTypeFuture,
                    leaseTermFuture).get();
        } catch (Exception e) {
            e.printStackTrace();
        }
        return resultVo;
    }
}
总结:

通过异步编排实现多任务并行的场景的执行优化,手动配置线程池可以让我们根据具体业务,更好的设置线程池的参数。

 参考:CompletableFuture的介绍和相关API

  • 3
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
使用`CompletableFuture`可以方便地实现异步执行多个任务,并在所有任务完成后进行处理。下面是一个示例代码: ```java import java.util.concurrent.CompletableFuture; public class CompletableFutureExample { public static void main(String[] args) { // 创建多个CompletableFuture对象 CompletableFuture<String> task1 = CompletableFuture.supplyAsync(() -> "Task 1"); CompletableFuture<String> task2 = CompletableFuture.supplyAsync(() -> "Task 2"); CompletableFuture<String> task3 = CompletableFuture.supplyAsync(() -> "Task 3"); // 执行所有任务并等待完成 CompletableFuture<Void> allTasks = CompletableFuture.allOf(task1, task2, task3); // 在所有任务完成后进行处理 allTasks.thenRun(() -> { try { // 获取任务的结果 String result1 = task1.get(); String result2 = task2.get(); String result3 = task3.get(); System.out.println("Task 1 result: " + result1); System.out.println("Task 2 result: " + result2); System.out.println("Task 3 result: " + result3); // 继续后续的操作 // ... } catch (Exception e) { e.printStackTrace(); } }); // 等待所有任务完成 allTasks.join(); } } ``` 在上述代码,我们创建了三个`CompletableFuture`对象:`task1`、`task2`和`task3`,它们代表了三个异步任务。然后,我们使用`CompletableFuture.allOf()`方法将这三个任务组合成一个新的`CompletableFuture`对象`allTasks`,该对象将在所有任务完成后触发。在`allTasks.thenRun()`,我们定义了所有任务完成后的处理逻辑,包括获取每个任务的结果并进行处理。最后,我们调用`allTasks.join()`方法等待所有任务完成。 使用`CompletableFuture`可以更灵活地处理异步任务的结果和异常,还可以通过方法链的方式组合多个任务。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值