前面我们对Lambda表达式已经有了初步地理解,下面我们来介绍下Lambda的语法。
一、Lambda表达式组成
Lambda表达式是从函数式接口定义衍生出来的,类似于一般方法,它也有自己的入参、执行体和一个返回值,或许还可能抛出异常。如果从方法的角度来理解,我们可能更容易抓住Lambda语法的要点。
Lambda式用“->”来分成两个部分,箭头左边的部分是Lambda表达式的入参,箭头右边的部分是Lambda表达式要执行的内容,包括返回值,基本格式如下:
(Input argements) -> { body }
先来看几个例子:
Example 1:
( ) -> System.out.println("Hello, Lambdas!");
在这个例子中,左边表示入参的地方是一对空括号,表示该Lambda表达式不需要入参。右边的执行体表示Lambda的业务逻辑,现在只有一行打印代码,表示返回空并且没有任何异常声明。
注:如果执行体是一行简单的语句或表达式时,则箭头右边包含执行体的"{ }"可以省略。
Example 2:
(Trade t)-> t.getCountry().equals("China")
该例中,箭头左边的入参是一个参数,右边是对应要执行的语句。此例还可以写成下面格式:
t -> t.getCountry().equals("China")
注: 只有一个入参的Lambda表达式,可以省略箭头左边的"( )"。
此处简写后,有个问题,即编译器如何知道参数 t的类型呢?这个是因为前面已经说过,任何一个Lambda表达式都是从相应的函数式接口定义衍生的,编译器可以根据上下文来推断出相应的参数对应的类型,所以我们可以忽略参数的类型。后面我们将会详细介绍Lambda表达式的类型。
函数式接口: 函数式接口就是一个有且仅有一个抽象方法的接口。
二、Lambda表达式的返回值
对于Lambda表达式的返回值,在一些情况下,我们需要显式的声明,但是有些情况下,可以隐式的声明,此时return关键字可以省略。来看一个例子:
t -> t.isCancelled()
在上面的例子中,我们并没有在执行体中声明return值。因为这是一个简单的表达式,其返回值被隐式地返回给调用者。即当执行体是一个简单的表达式或语句时,可以省略相应的return语句。
但是,如果执行体是一个包含各种控制语句的复杂的代码块时,就必须要显示地带上return语句。
t -> {
return t.isCancelled();
}
现在由于加上了"{ }",就表示不再是简单的语句或表达式了,此时就必须显示地带上return关键字。
三、Lambda表达式的类型
前面已经说过,Lambda表达式的格式取决于相应函数式接口的定义。我们首先从一个简单的"Hello world!"例子来说明:
interface Greeting {
void sayHello(String name);
}
Greeting是一个函数式接口,包含一个抽象的方法sayHello,入参是一个String,返回值是空。现在我们来看下如何定义一个Greeting类型的Lambda表达式。
首先因为sayHello方法的入参是String类型,所以Lambda表达的左边也应当是String类型,写法如下:
(String msg)
右边的逻辑是应用给定的入参,根据接口的定义,应该是打印一条信息到控制台。代码如下:
System.out.println("Hello, " + msg);
现在我们把左右两边用"->"分割,就形成了一个Lambda表达式:
(String msg) -> System.out.println("Hello, " + msg);
再来用传统的方法实现该接口看下:
Greeting oldGreeting = new Greeting() {
@Override
public void sayHello(String name) {
System.out.println("Hello, " + name);
}
};
对比两种写法,可以看出,把传统方式写法的入参加上执行体即组成了 Lambda表达式。
既然是表达式,就可以赋值给一个变量,当然Lambda表达式也可以赋值给某个变量,这个变量要求必须是对应的函数接口类型:
Greeting greetingLambda = (String msg) -> System.out.println("Hello, " + msg);
四、实例
最后我们再用一个稍微复杂的例子来加强下对Lambda是语法的理解,这次我们带两个输入参数并且返回一个对象。
public interface TradeMerge {
Trade merge(Trade t1, Trade t2);
}
该接口是合并两个订单形成一个新的订单,由此定义了一个单独的抽象方法"merge",接收两个入参,并且有返回值。
根据同样的方法,我们来实现这个对应的Lambda表达式。首先因为"merge"方法有两个入参,则Lambda表达式箭头左边的参数集合也应该有两个参数:
(Trade t1, Trade t2)
右边的执行体需要根据两个订单生成一个新的订单,实现如下:
Trade mergedTrade = new Trade();
mergedTrade.setQuantity(t1.getQuantity() + t2.getQuantity());
return mergedTrade;
现在把左右用 “->”组合在一起,则该Lambda式如下:
(Trade t1, Trade t2) -> {
Trade mergedTrade = new Trade();
mergedTrade.setQuantity(t1.getQuantity() + t2.getQuantity());
return mergedTrade;
}
或者更简单,我们可以省略参数类型:
(t1, t2) -> {
Trade mergedTrade = new Trade();
mergedTrade.setQuantity(t1.getQuantity() + t2.getQuantity());
return mergedTrade;
}
我们也可以把该表达式赋值给一个变量:
TradeMerge mergeTrade = (t1, t2) -> {
Trade newTrade = new Trade();
newTrade.setQuantity(t1.getQuantity() + t2.getQuantity());
return newTrade;
};