参考地址:https://www.jianshu.com/p/6f3ee90ab7d3

示例:

public static void main(String[] args) throws InterruptedException, ExecutionException {

        CompletableFuture<String>  cf1 = new CompletableFuture<>();
new Thread(() -> {
// 模拟执行耗时任务
System.out.println("task doing...");
try {
Thread.sleep(3000);
} catch (Exception e) {
cf1.completeExceptionally(e);
}
// 告诉completableFuture任务已经完成
cf1.complete("3");
}).start(); CompletableFuture<String> cf2 = new CompletableFuture<>();
new Thread(() -> {
// 模拟执行耗时任务
System.out.println("task doing...");
try {
Thread.sleep(1000);
} catch (Exception e) {
cf2.completeExceptionally(e);
}
// 告诉completableFuture任务已经完成
cf2.complete("1");
}).start(); System.out.println(cf1.get());
System.out.println(cf2.get()); }

在spring cloud微服务中调用分别几个其他微服务中的服务接口,放置单线程进行调用,导致接口超时的问题,应用Completablefuture 解决:

@RestController
public class TenTenementApiImpl implements ITenTenementApi{ @Autowired
private SysTenementService sysTenementService; @Autowired
private SysTenementConfigService sysTenementConfigService; @Autowired
FeignTenBrandClient brandClient; @Autowired
FeignTenDealerClient tenDealerClient; @Autowired
FeignTenMemberClient tenMemberClient; @Autowired
TenSecurityCodeCountClient tenSecurityCodeCountClient; /**
* 租户首页 统计
* 企业概况
*
* @return
*/
@Override
public AjaxResult<TenStatisticalBean> tenStatistical() { AjaxResult<TenStatisticalBean> res = new AjaxResult<>();
TenementUser tenementUser = RequestData.TENEMENT_USER.get(); //获取租户的企业参数 获取 租户生成码 展示 内码/外码
SysTenementConfig sysTenementConfig = sysTenementConfigService.findByTenementId(tenementUser.getTenementId());
if (sysTenementConfig != null){ String tenJson = JSON.toJSONString(tenementUser);
TenStatisticalBean bean = new TenStatisticalBean();
bean.setTid(tenementUser.getTenementId());
try { //商品统计,调用ms-goods服务
CompletableFuture<Integer> cf1 = new CompletableFuture<>();
new Thread(() -> { System.out.println("异步商品统计---->");
Integer goodsCount = 0;
try {
AjaxResult<Integer> goodsRes = brandClient.countGoodsByTid(tenJson);
if (goodsRes.isSuccess()){
goodsCount = goodsRes.getObj();
}
} catch (Exception e) {
goodsCount = null;
}
// 告诉completableFuture任务已经完成
cf1.complete(goodsCount);
}).start(); //经销商统计,调用ms-dealer服务
CompletableFuture<Integer> cf2 = new CompletableFuture<>();
new Thread(() -> { System.out.println("异步经销商统计---->");
Integer dealerCount = 0;
try {
AjaxResult<Integer> dealerRes = tenDealerClient.countDealerByTid(tenJson);
if (dealerRes.isSuccess()){
dealerCount = dealerRes.getObj();
}
} catch (Exception e) {
dealerCount = null;
}
// 告诉completableFuture任务已经完成
cf2.complete(dealerCount);
}).start(); //会员统计,调用ms-member服务
CompletableFuture<Integer> cf3 = new CompletableFuture<>();
new Thread(() -> { System.out.println("异步会员统计---->");
Integer memberCount = 0;
try {
AjaxResult<Integer> memberRes = tenMemberClient.countMemberByTid(tenJson);
if (memberRes.isSuccess()){
memberCount = memberRes.getObj();
}
} catch (Exception e) {
memberCount = null;
}
// 告诉completableFuture任务已经完成
cf3.complete(memberCount);
}).start(); //防伪码统计,调用ms-code服务
CompletableFuture<SecurityCodeCountBean> cf4 = new CompletableFuture<>();
new Thread(() -> { System.out.println("异步防伪码统计---->");
SecurityCodeCountBean securityCodeCount = new SecurityCodeCountBean();
try {
AjaxResult<SecurityCodeCountBean> scRes = tenSecurityCodeCountClient.countScCodeByTid(tenJson);
securityCodeCount = scRes.getObj();
if (scRes.isSuccess() && securityCodeCount != null){ }
} catch (Exception e) {
securityCodeCount = null;
}
// 告诉completableFuture任务已经完成
cf4.complete(securityCodeCount);
}).start(); bean.setGoodsCount(cf1.get());
bean.setDealerCount(cf2.get());
bean.setMemberCount(cf3.get());
SecurityCodeCountBean securityCodeCount = cf4.get();
if (securityCodeCount != null){
//设置码号生成总数
bean.setScCodeCount(securityCodeCount.getScCode());
//设置历史库存
bean.setHistoryStock(securityCodeCount.getHistoryStock());
//即时库存
bean.setImmediateStock(securityCodeCount.getImmediateStock());
//经销商库存
bean.setDealerStock(securityCodeCount.getDealerStock());
//已出售
bean.setSellCount(securityCodeCount.getSellCount());
//在途
bean.setOnWayCount(securityCodeCount.getOnWayCount());
} res.initTrue(bean);
} catch (InterruptedException e) {
throw new LunaException(e.getMessage(), LunaResultBean.ERROR_BUSINESS);
} catch (ExecutionException e) {
throw new LunaException(e.getMessage(), LunaResultBean.ERROR_BUSINESS);
}
}else {
res.initFalse("租户未配置企业参数",LunaResultBean.ERROR_BUSINESS);
}
return res;
}

附接收的实体:

public class TenStatisticalBean {

    private String tid;//租户ID

    private Integer goodsCount;//商品统计

    private Integer dealerCount;//经销商统计

    private Integer memberCount;//会员统计

    private Long scCodeCount;//码号生成统计【此处及以下防伪码相关字段 都只展示  租户配置的产品码类型  内码量/外码量】

    private Long historyStock;//历史库存    总共生成码 入 数据库的总量

    private Long immediateStock;//即时库存  租户仓库 的库存数量  发货给经销商-   经销商退货给租户+

    private Long onWayCount;//在途统计

    private Long dealerStock;//经销商库存    经销商发货-  经销商收货+  出售- 出售的退货+

    private Long sellCount;//出售统计   所有出售的+ 退货给经销商-

    public Long getOnWayCount() {
return onWayCount;
} public void setOnWayCount(Long onWayCount) {
this.onWayCount = onWayCount;
} public String getTid() {
return tid;
} public void setTid(String tid) {
this.tid = tid;
} public Integer getGoodsCount() {
return goodsCount;
} public void setGoodsCount(Integer goodsCount) {
this.goodsCount = goodsCount;
} public Integer getDealerCount() {
return dealerCount;
} public void setDealerCount(Integer dealerCount) {
this.dealerCount = dealerCount;
} public Integer getMemberCount() {
return memberCount;
} public void setMemberCount(Integer memberCount) {
this.memberCount = memberCount;
} public Long getScCodeCount() {
return scCodeCount;
} public void setScCodeCount(Long scCodeCount) {
this.scCodeCount = scCodeCount;
} public Long getHistoryStock() {
return historyStock;
} public void setHistoryStock(Long historyStock) {
this.historyStock = historyStock;
} public Long getImmediateStock() {
return immediateStock;
} public void setImmediateStock(Long immediateStock) {
this.immediateStock = immediateStock;
} public Long getDealerStock() {
return dealerStock;
} public void setDealerStock(Long dealerStock) {
this.dealerStock = dealerStock;
} public Long getSellCount() {
return sellCount;
} public void setSellCount(Long sellCount) {
this.sellCount = sellCount;
}
}

【多线程】java多线程Completablefuture 详解【在spring cloud微服务之间调用,防止接口超时的应用】【未完成】的更多相关文章

  1. Spring Cloud微服务Sentinel+Apollo限流、熔断实战总结

    在Spring Cloud微服务体系中,由于限流熔断组件Hystrix开源版本不在维护,因此国内不少有类似需求的公司已经将眼光转向阿里开源的Sentinel框架.而以下要介绍的正是作者最近两个月的真实 ...

  2. Spring Cloud微服务限流之Sentinel+Apollo生产实践

    Sentinel概述 在基于Spring Cloud构建的微服务体系中,服务之间的调用链路会随着系统的演进变得越来越长,这无疑会增加了整个系统的不可靠因素.在并发流量比较高的情况下,由于网络调用之间存 ...

  3. Spring Cloud微服务学习笔记

    Spring Cloud微服务学习笔记 SOA->Dubbo 微服务架构->Spring Cloud提供了一个一站式的微服务解决方案 第一部分 微服务架构 1 互联网应用架构发展 那些迫使 ...

  4. Spring Cloud微服务系列文,服务调用框架Feign

    之前博文的案例中,我们是通过RestTemplate来调用服务,而Feign框架则在此基础上做了一层封装,比如,可以通过注解等方式来绑定参数,或者以声明的方式来指定请求返回类型是JSON.    这种 ...

  5. 如何优化Spring Cloud微服务注册中心架构?

    作者: 石杉的架构笔记 1.再回顾:什么是服务注册中心? 先回顾一下什么叫做服务注册中心? 顾名思义,假设你有一个分布式系统,里面包含了多个服务,部署在不同的机器上,然后这些不同机器上的服务之间要互相 ...

  6. Dubbo和Spring Cloud微服务架构比较

    Dubbo 出生于阿里系,是阿里巴巴服务化治理的核心框架,并被广泛应用于中国各互联网公司:只需要通过 Spring 配置的方式即可完成服务化,对于应用无入侵,设计的目的还是服务于自身的业务为主. 微服 ...

  7. 在阿里云容器服务上开发基于Docker的Spring Cloud微服务应用

    本文为阿里云容器服务Spring Cloud应用开发系列文章的第一篇. 一.在阿里云容器服务上开发Spring Cloud微服务应用(本文) 二.部署Spring Cloud应用示例 三.服务发现 四 ...

  8. Dubbo 和 Spring Cloud微服务架构 比较及相关差异

    你真的了解微服务架构吗?听听八年阿里架构师怎样讲述Dubbo和Spring Cloud微服务架构. 微服务架构是互联网很热门的话题,是互联网技术发展的必然结果.它提倡将单一应用程序划分成一组小的服务, ...

  9. 全链路实践Spring Cloud 微服务架构

    Spring Cloud 微服务架构全链路实践Spring Cloud 微服务架构全链路实践 阅读目录: 网关请求流程 Eureka 服务治理 Config 配置中心 Hystrix 监控 服务调用链 ...

随机推荐

  1. 进程一些命令pstree,ps,pstack,top

    1. pstree pstree以树结构显示进程$ pstree -p work | grep adsshd(22669)---bash(22670)---ad_preprocess(4551)-+- ...

  2. 次短路经(dijsktra)

    #include <cstdio>#include <cstring>#include <queue>#include <algorithm>#defi ...

  3. Vue进阶篇

    前引 今天是2018年12月30,虽不是2018年的最后一天,但是却是自己在2018年写的最后一篇博客了,昨天下班在地铁上闲来无事,翻起了关注的一些公众号发的技术博文,里面就提到写博客的重要性,其实这 ...

  4. Java学习(final、static关键词)

    final关键词 概念:final的意思为最终,不可变.final是个修饰符,它可以用来修饰类,类的成员,以及局部变量.不能修饰构造方法. 特点: 1.final修饰的类不可以被继承,但可以继承别的类 ...

  5. day5冒泡排序

    冒泡排序:是一种基础的算法,实现数据的排序,排序的原则是前一个与后一个进行比较,如果前面的值大则交换,否则不交换,多次循环每次把最大的数据循环至后面就能够完成所需. 上面的图是冒泡排序的原理,每次循环 ...

  6. c++ primer 4 数组和指针

    类比的思想学习数组和指针,c++提供类似于vector和迭代器的低级复合类型——数组和指针.与vector相似,数组可以保存某一种类型的一组对象:而他们的区别在于,数组的长度固定,数组一经创建就不允许 ...

  7. vuejs学习——vue+vuex+vue-router项目搭建(二)

    前言 最近比较忙,所有第二章发布晚了,不好意思各位. vuejs学习——vue+vuex+vue-router项目搭建(一) 中我们搭建好了vue项目,我相信大家已经体验了vue其中的奥妙了,接下来我 ...

  8. Map 的四种遍历方式

    Map 的四种遍历方式 import java.util.HashMap; import java.util.Iterator; import java.util.Map; public class ...

  9. HTML5实战与剖析之原生拖拽(一拖拽历史概述)

    提起拖拽,我就想起了在JavaScript培训的时候一个非常好玩的效果,那就是拖拽了.可以用鼠标任意拖拽着一个物体到任何你想去的地方. 最早拥有JavaScript拖拽功能的是IE4浏览器.当时,网页 ...

  10. mongoDB学习第二天之常用方法

    mongoDB LIMIT 和 SKIP 方法 db.colName.find().limit(num)  # limit 方法接收一个数字参数,该参数指定读取的记录条数 (db.colName.fi ...