在 Java 8 lambda 语法中,大括号何时是可选的?

2022-09-02 12:11:57

我意识到Java 8 lambda实现可能会发生变化,但是在lambda构建b39中,我发现只有当lambda表达式返回非void类型时,才能省略大括号。例如,这会编译:

public class Collections8 {
        public static void main(String[] args) {
                Iterable<String> names = Arrays.asList("Alice", "Bob", "Charlie");
                names.filter(e -> e.length() > 4).forEach(e -> { System.out.println(e); });
        }
}

但是像这样删除大括号:

names.filter(e -> e.length() > 4).forEach(e -> System.out.println(e));

给出错误

Collections8.java:6: error: method forEach in interface Iterable<T> cannot be applied to given types;
        names.filter(e -> e.length() > 4).forEach(e -> System.out.println(e));
                                         ^
  required: Block<? super String>
  found: lambda
  reason: incompatible return type void in lambda expression
  where T is a type-variable:
    T extends Object declared in interface Iterable

谁能解释一下这里发生了什么?


答案 1

当 lambda 主体是单个表达式或 void 方法调用时,可以省略大括号。每个表达式的计算结果都为一个值,因此不能为 void。

如果 lambda 的主体是一个语句块(例如,一系列计算后跟一个语句),或者 lambda 没有值(即具有返回类型)并且不是单个 void 方法调用,则必须使用块形式,这需要方括号。returnvoid

在块式 lambda 中,如果对值进行 ed,则所有可能的代码路径都必须是值或 .returnreturnthrowThrowable


答案 2

这只是:EG(大部分)已经对语法做出了决定。

在考虑了许多替代方法之后,我们决定基本上采用 C# 语法。我们可能仍然会进一步考虑细节(例如,细箭头与胖箭头,特殊nilary形式等),并且尚未就方法参考语法做出决定。

C# 语法为:

lambda = ArgList Arrow Body
ArgList = Identifier
           | "(" Identifier [ "," Identifier ]* ")"
           | "(" Type Identifier [ "," Type Identifier ]* ")"
Body = Expression
           | "{" [ Statement ";" ]+ "}"

表达式的计算结果为某物,Java 中不能有 void 表达式。这是一个声明,因此您需要绕过它。{}

http://mail.openjdk.java.net/pipermail/lambda-dev/2011-September/003936.html


推荐