编程语言
首页 > 编程语言> > 为什么不传递java中的每个方法在传递Function类型参数而不是Consumer时抛出异常?

为什么不传递java中的每个方法在传递Function类型参数而不是Consumer时抛出异常?

作者:互联网

参见英文答案 > Why do Consumers accept lambdas with statement bodies but not expression bodies?                                    3个
>            Lambda ‘special void-compatibility rule’ – statement expression                                    3个
>            Why does a Java method reference with return type match the Consumer interface?                                    2个
为什么java中的forEach方法在传递Function类型参数而不是Consumer时不显示编译器错误?这两行都为流中的每个元素返回一个布尔值,但只有第二行得到编译错误?对于这种情况,lambda表达式是否还有其他属性?

这是我的代码:

Stream.of(1,2,3,4).forEach(a->a.equals(1));//line 1

Stream.of(1,2,3,4).forEach(a->{return a.equals(1);});//line 2

解决方法:

关于第一行的工作原理:the specification中有一个解释:

Generally speaking, a lambda of the form () -> expr, where expr is a statement expression, is interpreted as either () -> { return expr; } or () -> { expr; }, depending on the target type.

以上是以下示例(很好的巧合,这与您的示例非常相似):

// Consumer has a void result
java.util.function.Consumer<String> c = s -> list.add(s);

这只是意味着编译器会忽略表达式的返回类型,就好像你的代码只是这样(对于void方法有效):

Stream.of(1, 2, 3, 4).forEach(a -> {
     a.equals(1);
});

关于第二行,the spec说:

A block lambda body is void-compatible if every return statement in the block has the form return;.

但是,在您的情况下,{return a.equals(1);}不符合此规则. Void方法不返回值.

理解这一点的一个简单方法是考虑编译器应用方法体验证规则(这样主体必须与声明public void accept(T t)兼容) – 如tutorial中所述

标签:functional-interface,java,java-8,lambda
来源: https://codeday.me/bug/20190731/1586527.html