多线程使用,带返回值的线程池

业务需求拿到多个接口返回的信息集合,并通过判断对比整合数据,考虑可能并发比较高,所以引用线程池来完成; 可返回值的任务必须实现Callable接口。

执行Callable任务后,可以获取一个Future的对象,在该对象上调用get就可以获取到Callable任务返回的Object了。 创建一个线程池,在线程中调用其他接口来实现业务需求,并把结果返回,例子如下:

// 创建一个线程池
    ExecutorService executorService = Executors.newFixedThreadPool(3);
    List<Future> list = new ArrayList<>();
    // 执行任务并获取Future对象
    String requestId = LogUtil.getCurrentRequestId();
    RequestAttributes requestAttributes = RequestContextHolder.getRequestAttributes();
    // 多线程实现商品状态线程
    Future goodsStautsFuture =
        executorService.submit(
            new Callable<String>() {
              @Override
              public String call() throws Exception {
                LogUtil.setRequestId(requestId);
                RequestContextHolder.setRequestAttributes(requestAttributes);
                GetCartOrderOhterMsgResultDto result =
                    orderPlaceOrderPageApiExecutorService.checkGoodsStauts(
                        listGoodsStatusParamsDto);
                result.setCode(1);
                logger.info("校验下单检查商品状态:result: {}", StringUtil.getJsonString(result));
                return StringUtil.getJsonString(result);
              }
            });
    list.add(goodsStautsFuture);
    // 获取商品价格信息
    Future checkSearchPriceFuture =
        executorService.submit(
            new Callable<String>() {
              @Override
              public String call() throws Exception {
                LogUtil.setRequestId(requestId);
                RequestContextHolder.setRequestAttributes(requestAttributes);
                GetCartOrderOhterMsgResultDto result =
                    orderPlaceOrderPageApiExecutorService.checkSearchPrice(
                        listPricePramasDto,
                        insertCartOrderParamDto,
                        Integer.valueOf(customerId.toString()));
                result.setCode(2);
                logger.info("校验获取商品价格:result: {}", StringUtil.getJsonString(result));
                return StringUtil.getJsonString(result);
              }
            });
    list.add(checkSearchPriceFuture);
    // 获取其他订单信息
    Future checkOtherMsgFuture =
        executorService.submit(
            new Callable<String>() {
              @Override
              public String call() throws Exception {
                LogUtil.setRequestId(requestId);
                RequestContextHolder.setRequestAttributes(requestAttributes);
                GetCartOrderOhterMsgResultDto result =
                    orderPlaceOrderPageApiExecutorService.checkOtherMsg(
                        insertCartOrderParamDto, Integer.valueOf(customerId.toString()));
                result.setCode(3);
                logger.info("校验获取其他订单信息:result: {}", StringUtil.getJsonString(result));
                return StringUtil.getJsonString(result);
              }
            });
    list.add(checkOtherMsgFuture);
    // 关闭线程池
    executorService.shutdown();

	for (Future f : list) {
        String msg=f.get().toString();
     }

在分布式程序中,调用其他功能需要用到requestid等信息,以前我们service层要想使用域对象必须先从Controller层中获取再传递过去,在线程池中会出现无法获取FeignClient的情况,这里做了特殊处理: 执行线程之前先获取 String requestId = LogUtil.getCurrentRequestId(); RequestAttributes requestAttributes = RequestContextHolder.getRequestAttributes(); 在线程中赋值 LogUtil.setRequestId(requestId); RequestContextHolder.setRequestAttributes(requestAttributes);

线程创建的几种方式:

  1. 继承Thread类
  2. 实现Runnable接口
  3. 匿名类的方式
经验分享 程序员 微信小程序 职场和发展