我创建了一个函数来过滤多个谓词,并对它们执行逻辑AND:
@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为什么会这样呢?

对于infos,我使用Eclipse的露娜版本1。
发布于 2015-01-05 17:09:11
对于编译器来说,这是一个很小的例子。为了确定它是否应该将参数的varargs包装应用到数组中,或者只是传递一个数组,它需要知道最后一个参数的类型,然而,对于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);
}这将是一个可接受的变通方法,因为它不需要在调用端进行任何更改,同时提高了单参数情况下的效率。
请注意,您的varargs方法允许零参数,但如果以这种方式调用,将会失败。因此,您应该添加另一个重载:
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);
}发布于 2015-01-05 09:12:11
Netbeans和Eclipse在解析Java中的lambda表达式方面都有许多错误。它们正在慢慢得到修复,但在修复之前,我找到的最好的解决方法是: 1.声明类型2.如果不起作用,使用块3.如果不起作用,创建一个实现谓词/函数的匿名对象,等等。
这会使您的代码更加混乱,但在许多情况下它们是必要的。
发布于 2017-04-25 17:16:30
有时Eclipse需要对所有项目进行彻底的清理和重建,这样问题就会消失。
https://stackoverflow.com/questions/27771488
复制相似问题