
本文探讨了如何使用 Java Stream API 从 List
使用对象而非 Map
当处理具有固定键值对结构的数据时,使用 HashMap 并非最佳实践。更好的方法是创建一个自定义对象,该对象具有与 Map 中键对应类型的属性。例如,对于 {"Start":"A", "End":"B","Length":5} 这样的数据结构,可以创建一个如下的 Java 16 record:
public record Foo(String start, String end, int length) {}使用对象可以提高代码的可读性和类型安全性,避免了从 HashMap 中获取 Object 类型值时需要进行类型转换的问题。
查找单个最大元素
如果只需要找到一个具有最大 length 值的 Foo 对象,可以使用 Stream.max() 或 Collections.max() 方法。这两个方法都需要一个 Comparator 实例作为参数,用于比较对象。
立即学习“Java免费学习笔记(深入)”;
使用 Stream API:
Listfoos = // 初始化列表 Foo max = foos.stream() .max(Comparator.comparingInt(Foo::length)) // 产生 Optional .orElseThrow();
这段代码首先将 List
使用 Collections.max():
Listfoos = // 初始化列表 Foo max = Collections.max(foos, Comparator.comparingInt(Foo::length));
Collections.max() 方法直接在集合上查找最大值,无需转换为 Stream。
查找所有具有最大长度的元素
如果需要找到所有具有最大 length 值的 Foo 对象,则需要进行额外的处理。以下提供了两种方法:
使用 groupingBy() 收集器:
Listfoos = // 初始化列表 List max = foos.stream() .collect(Collectors.groupingBy(Foo::length)) // Map > .entrySet().stream() // Stream >> .max(Map.Entry.comparingByKey()) // Optional >> .map(Map.Entry::getValue) // Optional > .orElse(Collections.emptyList());
这段代码首先使用 Collectors.groupingBy(Foo::length) 将 List
使用三参数 collect() 方法:
Listfoos = // 初始化列表 List max = foos.stream() .collect( ArrayList::new, (List l, Foo f) -> { if (!l.isEmpty() && l.get(0).length() < f.length()) l.clear(); if (l.isEmpty() || l.get(0).length() == f.length()) l.add(f); }, (l, r) -> { if (l.get(0).length() < r.get(0).length()) l.clear(); if (l.isEmpty() || l.get(0).length() == r.get(0).length()) l.addAll(r); } );
这种方法使用 Stream.collect() 的三参数版本,避免了创建中间 Map。第一个参数 ArrayList::new 提供了一个 Supplier,用于创建一个新的 ArrayList 作为累加器。第二个参数是一个 BiConsumer,用于将 Stream 中的元素累加到累加器中。第三个参数是一个 BiConsumer,用于合并两个累加器。
在这个例子中,BiConsumer 的逻辑是:如果当前元素 f 的 length 值大于累加器 l 中第一个元素的 length 值,则清空累加器 l;如果累加器 l 为空或者当前元素 f 的 length 值等于累加器 l 中第一个元素的 length 值,则将当前元素 f 添加到累加器 l 中。
总结
本文介绍了如何使用 Java Stream API 从 List










