繁体   English   中英

在Java 8中应用一系列过滤器是否有优雅的方法?

[英]Is there an elegant way of applying an array of filters in Java 8?

有没有办法使用reduce或类似的方法将一组过滤器应用于对象数组?

现在,我有这个:

private void applyManyFilters(long[] initialData, LongPredicate... filters) {
    LongStream stream = Arrays.stream(initialData);
    for (LongPredicate filter : filters) {
      stream = stream.filter(filter);
    }
    long[] dataToUse = stream.toArray();

    // Use filtered data
}

更优雅的解决方案看起来像这样:

private void applyManyFilters(long[] initialData, LongPredicate... filters) {
    long[] dataToUse = Arrays.stream(filters).reduce(Arrays.stream(initialData), (a, b) -> a.filter(b)).toArray();

    // Use filtered data here
}

当然,上面的例子没有编译,因为要reduce两个参数都需要是LongPredicate ,而不是LongStream 我很好奇上面的代码是否可以写成一行?

(PS为了清楚起见,这是一个关于Java 8如何工作的问题,而不是关于样式的问题。我同意单行方法几乎不像循环一样可读。)

我相信这就是你要找的东西:

private void applyManyFilters(long[] initialData, LongPredicate... filters) {
    long[] dataToUse = Arrays.stream(initialData)
                             .filter(v -> Arrays.stream(filters)
                                                .allMatch(f -> f.test(v)))
                             .toArray();
    // Use filtered data
}

UPDATE

合并其他两个答案,您还可以创建一个帮助方法来组合过滤器数组。 你可以在helper类中使它成为一个很好的实用方法,以便在代码中的任何地方重用,同时覆盖其他谓词( BiPredicateDoublePredicateIntPredicatePredicate )。

private void applyManyFilters(long[] initialData, LongPredicate... filters) {
    long[] dataToUse = Arrays.stream(initialData)
                             .filter(combine(filters))
                             .toArray();
    // Use filtered data
}
public static LongPredicate combine(LongPredicate... filters) {
    return Arrays.stream(filters)
                 .reduce(LongPredicate::and)
                 .orElse(x -> true);
}

使用reduce的解决方案是使用LongPredicate::and将谓词减少为单个谓词LongPredicate::and如下所示:

//prints 6
public static void main(String[] args) {
    applyManyFilters(new long[]{1, 2, 3, 4, 5, 6}, l -> l % 2 == 0, l -> l % 3 == 0);
}

private static void applyManyFilters(long[] initialData, LongPredicate... filters) {
    Arrays.stream(initialData).filter(
            Arrays.stream(filters).reduce(l -> true, LongPredicate::and)
    ).forEach(System.out::println);
}

要继续@Andreas发布的答案,你可以使它成为一个静态泛型函数,因此它更可重用:

public static <T> Stream<T> applyManyFilters(Stream<T> data, Predicate<T>... filters) {
    return data.filter(d -> Arrays.stream(filters).allMatch(f -> f.test(d)));
}

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM