Java8是否可以方便地实现以下要求,即

Map<X,Y> + Map<Y,Z> = Map<X,Z>

现在我的代码是
ArrayList<String> couponCodes = newArrayList("aaa", "bbb", "ccc");
// couponCode -- shopId
ArrayList<ShopCoupon> shopCoupons = newArrayList(new ShopCoupon("aaa", 1), new ShopCoupon("bbb", 2), new ShopCoupon("ccc", 3));
Map<String, Integer> couponCodeShopIdMap = shopCoupons.stream().collect(toMap(sc -> sc.getCouponCode(), sc -> sc.getShopId()));
// shopId -- name
ArrayList<Shop> shops = newArrayList(new Shop(1, "zhangsan"), new Shop(2, "lisi"), new Shop(3, "wangwu"));
Map<Integer, String> shopIdNameMap = shops.stream().collect(toMap(s -> s.getId(), s -> s.getName()));

//couponCode -- name
Map<String, String> couponCodeNameMap = couponCodes.stream().collect(toMap(c -> c, c -> shopIdNameMap.get(couponCodeShopIdMap.get(c))));
System.out.println(couponCodeNameMap);

我想知道是否有一些更方便的方式来实现此要求?

最佳答案

另一种方法是直接流式传输第一个地图的条目集,例如:

public static <X, Y, Z> Map<X, Z> join(Map<X, Y> left, Map<Y, Z> right) {
    return left.entrySet()
               .stream()
               .collect(toMap(Map.Entry::getKey, e -> right.get(e.getValue())));
}

唯一需要处理的是,当左侧映射中存在一个值而不是右侧映射中的对应键时,您将拥有一个NullPointerException,因为不允许值映射器函数返回null。在这种情况下,您可以使用getOrDefault提供要与e -> right.getOrDefault(e.getValue(), defaultValue)关联的非空默认值,或者如果您不希望在结果映射中包含该映射,请通过过滤不存在​​的映射来模仿toMap收集器。
public static <X, Y, Z> Map<X, Z> join(Map<X, Y> left, Map<Y, Z> right) {
    return left.entrySet().stream().collect(
        HashMap::new,
        (m, e) -> {
            //if you still want the mapping k -> null,
            //you can just use put directly
            Z value = right.get(e.getValue());
            if(value != null) m.put(e.getKey(), value);
        },
        Map::putAll);
}

10-04 14:07