几天前,我开始重构一些代码以使用新的Java 8 Streams库。不幸的是,我在使用一个方法执行Stream :: map时遇到编译时错误,该方法被声明为抛出一个进一步指定为RuntimeException的泛型E.
有趣的是,当我切换到使用方法引用时,编译时错误就消失了。
这是一个错误,还是我的方法引用不等同于我的lambda表达式?
(另外,我知道我可以用Parameter :: execute替换p-> p.execute(foo)。我的实际代码有execute方法的附加参数。)
错误消息
Error:(32, 43) java: unreported exception E; must be caught or declared to be thrown
代码
import java.util.ArrayList;
import java.util.List;
public class JavaBugTest
{
interface AbleToThrowException<E extends Exception>
{
}
interface Parameter {
public <E extends Exception> Object execute(AbleToThrowException<E> algo) throws E;
}
interface ThrowsRuntimeException extends AbleToThrowException<RuntimeException>
{
}
static ThrowsRuntimeException foo;
public static Object manualLambda(Parameter p)
{
return p.execute(foo);
}
public static void main(String[] args)
{
List<Parameter> params = new ArrayList<>();
params.stream().map(p -> p.execute(foo)); // Gives a compile time error.
params.stream().map(JavaBugTest::manualLambda); // Works fine.
}
}
系统设置
答案 0 :(得分:3)
一个非常简单的解决方案是明确地为Parameter#execute(..)
提供一个类型参数。
params.stream().map(p -> p.<RuntimeException>execute(foo)); // Gives a compile time error.
如果没有显式类型参数,JDK编译器似乎无法从调用上下文中推断出类型参数,尽管它应该如此。这是一个bug,应该这样报告。我现在已经报告了这个问题,并且在我有这些问题后会用新的细节更新这个问题。