[英]The target type of this expression must be a functional interface
我创建了一个函数来过滤多个谓词,我为它们执行逻辑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 Luna版本1。
这是编译器的一个极端情况。 为了确定是否应该将参数的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);
}
这将是一个可接受的解决方案,因为它不需要在呼叫方面进行任何更改,同时同时提高单arg情况的效率。
请注意,您的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);
}
Netbeans和Eclipse在解析Java中的lambda表达式方面存在许多缺陷。 它们正在慢慢得到修复,但是,直到它们为止,我发现的最佳解决方法是:1。声明类型2.如果不起作用,使用块3.如果不起作用,则创建一个实现的匿名对象谓词/功能等
这些会使您的代码更加混乱,但在许多情况下它们都是必需的。
有时Eclipse需要一个良好的清理和重建所有项目,问题就会消失。
声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.