并发编程与视图——简单方法返回异步 Map<Key, Data>

Future 的基本用法与Maps#transformValue

本文所指的Future 适用于 Future 类型或者其子类如 CompletableFuture或者其子接口 ListenableFuture。

需求描述

我们在业务开发中常常遇到这种场景,执行业务时需要获取一些异步数据,常见的优化是在请求尽量早的位置且再业务处理前进行数据预加载,这些预加载数据类型为Future。

这些数据可以组成集合类 List<Future>,更常见的集合类型是 Map<String, Future<Data>>

实际上,对于后续使用方来说,其可能更愿意接受使用的是不带Future的版本。可以类比 Guava Cache 实现,获取的数据为真实类型而不是 Future 包装类型。

Guava 与懒计算视图

对于Map来说,可以结合Guava提供的Map#transform方法,添加懒计算。这种实现稍微容易理解一点,可以不使用 monadic 方法,monadic 方法指的是

  1. map: CompletableFuture#thenApply, ListenableFuture##transform
  2. flatMap: Completable#thenCompose, ListenableFuture##transformAsync

以比价平台简单举例,需要加载某东、某宝等商品价格数据。

ini 复制代码
// preload data
Map<String, Future<ProductInfo>> loadingAllData = xxx;
// 仅做简单举例,实际业务根据具体情况调整
Map<String, ProductInfo> allProducts = Maps.transformValues(loadingAllData, Futures#getUnchecked);
​
// 其他业务逻辑
​
// process data
var xxProduct = allProducts.get("xx");
List<ProductDetail> processedProducts = new ArrayList<>();
if (xxProduct != null) {
  processedProducts.add(processXxProduct(xxProduct));
}
processedProducts.stream().forEach(System.out::println);

同理,对于多值Map,可以使用 Mulitmaps#transformValues。

这里需要注意的是:

  1. 使用 allProducts 就像使用普通map一样,但是获取value值时实际上会执行 Futures#getUnchecked,后续使用方便。
  1. getUnchecked 可以改为带超时的get方法。一种是Future自带的get方法,另一种是 Guava Futures#withTimeout,这种超时可以保证超时返回超时异常,并且发送中断执行线程信号。
ini 复制代码
Map<String, ListenableFuture<Integer>> futureMap = Map.of("1", Futures.immediateFuture(1));
Map<String, Integer> simpleMap = Maps.transformValues(futureMap, f -> {
    FluentFuture<Integer> timedFuture = FluentFuture.from(Futures.withTimeout(f, Duration.ofMillis(250), timer))
            .catching(TimeoutException.class, e -> -1, MoreExecutors.directExecutor());
    return Futures.getUnchecked(timedFuture);
});
  1. 考虑happy-path,这种实现可以简化代码使用,避免出错,便于后续使用方使用。类似于 LoadingCache 。
  2. 异常场景需要注意,由于集合Future可能存在多个异常,需要做好异常处理。一般来说是做好日志,fallback,熔断等功能。
  3. 懒计算可以尽量保证取 value 时,数据加载已经进行了尽可能足够长的时间。假设数据加载需要300ms,超时时间设置为250ms,在提交加载后100调用Map#get 方法可以成功。
相关推荐
WeiXiao_Hyy2 分钟前
成为 Top 1% 的工程师
java·开发语言·javascript·经验分享·后端
苏渡苇8 分钟前
优雅应对异常,从“try-catch堆砌”到“设计驱动”
java·后端·设计模式·学习方法·责任链模式
团子的二进制世界15 分钟前
G1垃圾收集器是如何工作的?
java·jvm·算法
long31619 分钟前
Aho-Corasick 模式搜索算法
java·数据结构·spring boot·后端·算法·排序算法
rannn_1111 小时前
【苍穹外卖|Day4】套餐页面开发(新增套餐、分页查询、删除套餐、修改套餐、起售停售)
java·spring boot·后端·学习
灵感菇_1 小时前
Java HashMap全面解析
java·开发语言
qq_12498707531 小时前
基于JavaWeb的大学生房屋租赁系统(源码+论文+部署+安装)
java·数据库·人工智能·spring boot·计算机视觉·毕业设计·计算机毕业设计
短剑重铸之日1 小时前
《设计模式》第十一篇:总结
java·后端·设计模式·总结
若鱼19191 小时前
SpringBoot4.0新特性-Observability让生产环境更易于观测
java·spring
觉醒大王1 小时前
强女思维:着急,是贪欲外显的相。
java·论文阅读·笔记·深度学习·学习·自然语言处理·学习方法