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

her*_*ung 18 java lambda java-8

我意识到Java 8 lambda实现可能会发生变化,但在lambda build 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); });
        }
}
Run Code Online (Sandbox Code Playgroud)

但删除这样的大括号:

names.filter(e -> e.length() > 4).forEach(e -> System.out.println(e));
Run Code Online (Sandbox Code Playgroud)

给出了错误

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
Run Code Online (Sandbox Code Playgroud)

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

jpm*_*jpm 21

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

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

在块样式的lambda中,如果return编辑了值,则所有可能的代码路径必须是return值或throwa Throwable.

  • 几乎.应该澄清的是,表达式是一种语句(即所有表达式都是语句,但并非所有语句都是表达式).另请注意,块(由`{}`包围的代码)在语法上是单个语句.可以更清楚地说,lambda的主体可以包含单个语句,该语句可以是表达式或块语句. (2认同)

Ish*_*tar 11

这只是:EG(主要)决定语法.

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

C#语法是:

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

表达式求值为某事,您不能在Java中使用void表达式.这是一个声明,因此你需要{}它.

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