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

2022-09-03 15:31:18

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


答案 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);
}

这将是一个可以接受的解决方法,因为它不需要在调用端进行任何更改,同时提高单参数情况的效率。


请注意,您的 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);
}

答案 2

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

这些会使你的代码更混乱,但它们在很多情况下都是必要的。


推荐