在Java Stream API中,如果需要根据多个条件查找第一个满足条件的元素,直接使用多个filter和findFirst操作可能会遇到IllegalStateException,因为Stream只能被消费一次。以下将介绍如何解决这个问题,并提供一个通用的解决方案。
Stream与集合不同,它不是一个存储数据的容器,而是一个数据流。一旦Stream被消费(例如,通过findFirst、collect等终端操作),它就不能再被使用。尝试再次操作同一个Stream会抛出IllegalStateException。
要解决Stream的单次消费问题,可以将Stream转换为一个可重用的集合,例如List或Map。这样,就可以在集合上进行多次查找操作。
以下代码展示了如何将Stream转换为LinkedHashMap,并根据给定的键的优先级查找第一个存在的值:
import java.util.*;
import java.util.function.Function;
import java.util.stream.Collectors;
import java.util.stream.Stream;
public class StreamFindFirst {
private static T findBestValue(Stream stream, T... keys) {
Map map = stream.collect(Collectors.toMap(
Function.identity(),
Function.identity(),
(l, r) -> l,
LinkedHashMap::new
));
return Arrays.stream(keys)
.map(map::get)
.filter(Objects::nonNull)
.findFirst()
.orElse(null);
}
public static void main(String[] args) {
Stream stream1 = Stream.of("a", "b", "c", "d");
Stream stream2 = Stream.of("b", "c", "d", "e");
Stream stream3 = Stream.of("d", "e", "f", "g");
System.out.println(findBestValue(stream1, "a", "b", "c"));
System.out.println(findBestValue(stream2, "a", "b", "c"));
System.out.println(findBestValue(stream3, "a", "b", "c"));
}
} 代码解释:
irst().orElse(null):输出结果:
a b null
通过将Stream转换为集合,可以避免Stream的单次消费限制,实现多条件查找首个匹配项的需求。 使用LinkedHashMap可以保持Stream元素的顺序,并提供高效的查找性能。 这种方法不仅解决了问题,还提供了更通用的解决方案,可以轻松地根据不同的查找条件进行调整。