Java 8 根据条件应用流过滤器

2022-09-01 07:41:49

在Java 8中,有没有办法根据条件对流应用过滤器,

我有这个流

if (isAccessDisplayEnabled) {
     src = (List < Source > ) sourceMeta.getAllSources.parallelStream()
         .filter(k - > isAccessDisplayEnabled((Source) k))
         .filter(k - > containsAll((Source) k, substrings, searchString))
         .collect(Collectors.toList());
 } else {
     src = (List < Source > ) sourceMeta.getAllSources.parallelStream()
         .filter(k - > containsAll((Source) k, substrings, searchString))
         .collect(Collectors.toList());
 }

我正在添加过滤器

.filter(k - > isAccessDisplayEnabled((Source) k)))

基于 if-else 条件的流。有没有办法避免这种情况,因为如果有更多的过滤器出现,那么它将很难维护。

请让我知道


答案 1

一种方法是这样做

Stream<Source> stream = sourceMeta.getAllSources.parallelStream().map(x -> (Source)x);
if(isAccessDisplayEnabled) stream = stream.filter(s -> isAccessDisplayEnabled(s));
src = stream.filter(s - > containsAll(s, substrings, searchString))
            .collect(Collectors.toList());

另一个

 src = sourceMeta.getAllSources.parallelStream().map(x -> (Source)x)
     .filter(isAccessDisplayEnabled? s - > isAccessDisplayEnabled(s): s -> true)
     .filter(s - > containsAll(s, substrings, searchString))
     .collect(Collectors.toList());

在任一情况下,请注意在开始时执行一个类型转换如何简化整个流管道。

这两种解决方案都避免了对每个流元素的重新评估,但是,当此代码被证明是性能关键时,第二种解决方案依赖于JVM的内联功能。isAccessDisplayEnableds -> true


答案 2

您的条件与您的方法具有相同的名称。我假设你的意思是让这些不同,所以让我们假设它是这样的:

if (someCondition) {
    src = (List < Source > ) sourceMeta.getAllSources.parallelStream()
        .filter(k - > isAccessDisplayEnabled((Source) k))
        .filter(k - > containsAll((Source) k, substrings, searchString))
        .collect(Collectors.toList());
} else {
    src = (List < Source > ) sourceMeta.getAllSources.parallelStream()
        .filter(k - > containsAll((Source) k, substrings, searchString))
        .collect(Collectors.toList());
}

如果要删除 if/else,可以改为在第一个筛选器中执行检查:

src = (List < Source > ) sourceMeta.getAllSources.parallelStream()
    .filter(k - > !someCondition || isAccessDisplayEnabled((Source) k))
    .filter(k - > containsAll((Source) k, substrings, searchString))
    .collect(Collectors.toList());

在其他情况下,您获取所有内容并删除 isAccessDisplayEnabled() 方法调用,因此条件实际上是“如果 someCondition 为 false 或 isAccessDisplayEnabled(k)”。如果某些条件显示为 false,则跳过 isAccessDisplayEnabled() 检查。


推荐