我创建了一个函数来过滤多个谓词,我为它们执行逻辑与:
@SafeVarargs
public static <T> Stream<T> filter(Stream<T> source, Predicate<T>... predicates) {
return source.filter(Arrays.stream(predicates).reduce(predicates[0], Predicate::and));
}
调用时:
filter(IntStream.range(0, 10).boxed(), x -> x % 2 != 0, x -> x%3 == 0).forEach(System.out::println);
它工作正常并打印 3 和 9。但是当我传递单个谓词时,例如:
filter(IntStream.range(0, 10).boxed(), x -> x % 2 != 0).forEach(System.out::println);
我得到一个编译错误:
The target type of this expression must be a functional interface
这是为什么?
对于信息,我使用 Eclipse Luna 版本 1。
最佳答案
这是编译器的极端情况。为了确定它是否应该将参数的可变参数包装应用到数组中或只是传递一个数组,它需要知道最后一个参数的类型,但是,在 lambda 表达式的情况下,它需要调用的方法签名来确定类型。但很明显应该发生什么,因为 lambda 表达式永远不可能是数组类型,因此 javac
可以毫无问题地编译它。
一个可接受的解决方法是重载该方法:
@SafeVarargs
public static <T> Stream<T> filter(Stream<T> source, Predicate<T>... predicates) {
return source.filter(
Arrays.stream(predicates).reduce(predicates[0], Predicate::and));
}
public static <T> Stream<T> filter(Stream<T> source, Predicate<T> predicate) {
return source.filter(predicate);
}
这将是一个可以接受的解决方法,因为它不需要对调用方进行任何更改,同时提高了单参数情况下的效率。
请注意,您的可变参数方法允许零参数,但如果以这种方式调用将会失败。所以你应该要么,添加另一个重载:
public static <T> Stream<T> filter(Stream<T> source) {
return source;
}
或者使该方法对于零参数情况是安全的:
@SafeVarargs
public static <T> Stream<T> filter(Stream<T> source, Predicate<T>... predicates) {
return Arrays.stream(predicates).reduce(Predicate::and)
.map(source::filter).orElse(source);
}
关于java - 此表达式的目标类型必须是函数式接口(interface),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/27771488/