在 Java 中,方法引用可用于创建功能接口的实例。但是,在某些情况下,当方法引用的返回类型与 Consumer 接口的 Accept 方法不匹配时,可能会出现混乱。
考虑以下代码:
Consumer<String> lambda1 = s -> {}; Function<String, String> lambda2 = s -> s; Consumer<String> lambda3 = LambdaTest::consume; // but s -> s doesn't work! Function<String, String> lambda4 = LambdaTest::consume;
当您可能会预料到,将 lambda3 分配给 Consumer 应该会失败,因为 Consumer 方法的返回类型(String)与accept 方法的预期 void 类型不匹配。
然而,令人惊讶的是,lambda3 的分配成功了。这是因为设计决策允许以与调用方法相同的方式使方法适应功能接口。这意味着任何有返回值的方法都可以分配给 Consumer,而其返回值将被忽略。
当涉及到 lambda 表达式时,规则稍微复杂一些。 Lambda 表达式有两种形式:
仅当表达式求值为某个值时,第一种形式 ((args) -> expression) 才是值兼容的。如果没有代码路径尝试返回值,则第二种形式 ((args) -> { statements* }) 可以是 void 兼容的。
表达式 s -> s 不兼容 void,因为 s 不是语句。但是,具有副作用的表达式(例如方法调用)可以用作语句。这意味着像 s -> 这样的表达式s.toString() 和 s -> i 可以是 void 兼容的。
因此,在 LambdaTest::consume 的情况下,方法引用被分配给 Consumer 接口,因为 Consumer 方法可以被调用,并且它的返回值可以被忽略。即使返回类型不匹配,这种设计决策也能实现方法和功能接口之间的适应性。
以上是为什么可以将带返回值的方法引用分配给 Java 中的消费者接口?的详细内容。更多信息请关注PHP中文网其他相关文章!