此表达式的目标类型必须是功能接口

时间:2015-01-04 23:25:55

标签: java eclipse java-8 eclipse-luna

我创建了一个过滤多个谓词的函数,我为它们执行逻辑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

为什么会这样?

enter image description here 对于信息,我使用Eclipse Luna版本1。

3 个答案:

答案 0 :(得分:8)

这是编译器的一个极端情况。为了确定是否应该将参数的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);
}

答案 1 :(得分:2)

Netbeans和Eclipse在Java中解析lambda表达式方面存在许多错误。他们正在慢慢修复,但是,直到他们,我发现最好的解决方法是: 1.声明类型 2.如果不起作用,请使用块 3.如果不起作用,请创建一个实现谓词/函数等的匿名对象。

这会使您的代码更加混乱,但在许多情况下它们都是必需的。

答案 2 :(得分:0)

有时Eclipse需要一个好的ol&#39;清理和重建所有项目,问题就消失了。