Lambda 表达式是 Java SE8 推出的新功能,也是Java第一次引入函数式编程的尝试。
Lambda表达式格式
Lambda 表达式可以看做是一种匿名函数,但是它没有访问修饰符、返回值和名字。Lambda表达式由两部分构成,形式参数和方法体,中间用“->”
符号分隔。其中的形式参数类型能够进行自动推断,可以不写。当然在某些特殊情况下,形参类型也是不可缺少的。方法体可以是简单的表达式或者代码块,下面是一些例子:
// 1. 不需要参数,返回值为 5
() -> 5
// 2. 接收一个参数(数字类型),返回其2倍的值
x -> 2 * x
// 3. 接受2个参数(数字),并返回他们的差值
(x, y) -> x – y
// 4. 接收2个int型整数,返回他们的和
(int x, int y) -> x + y
// 5. 接受一个 string 对象,并在控制台打印,不返回任何值(看起来像是返回void)
(String s) -> System.out.print(s)
函数式接口
函数式接口
要理解Lambda表达式,首先要了解一种特殊的接口:函数式接口。什么是函数式接口呢?简单来说就是只包含一个抽象方法的接口。Java标准库中的java.lang.Runnable
和java.util.Comparator
就是典型的函数式接口。对于函数式接口,我们就可以使用Lambda表达式来代替用传统匿名类来创建实例对象。
我们以Runnable
接口为例,用传统匿名类的方式创建一个线程:
public void runThread() {
new Thread(new Runnable() {
public void run() {
System.out.println("Run!");
}
}).start();
}
上面传统匿名类方式中,我们可以看到,我们需要new
一个接口名称,接口内部还要附带这个接口抽象方法的实现。而如果我们使用Lambda表达式,则代码非常简洁:
public void runThreadUseLambda() {
new Thread(() -> {
System.out.println("Run!");
}).start();
}
通过上面代码我们可以看到,Lambda表达式在两方面做了简化:
- 首先不需要声明
Runnable
接口,因为这可以通过上下文推断出来 - 其次不需要再写一个
run
方法的实现,因为函数式接口中只有一个方法
java.util.function包
在Java SE8之前标准库中的函数式接口并不多。JavaSE8增加了java.util.function
包,里面都是可以在开发中只用的函数式接口。我们也可以自定义一个函数式接口,但最好在接口上使用@FunctionalInterface
注解标明这是一个函数式接口,以免团队其它成员错误地往接口添加新的方法,当然java.util.function
包中的所有接口都添加了@FunctionalInterface
注解。
下面代码使用函数式接口java.util.Function
接口实现的对列表map进行操作的方法,从代码中可以看出,如果使用函数式编程,代码看起来会非常简洁:
public class CollectionUtils {
public static List map(List input, Function processor) {
ArrayList result = new ArrayList();
for (T obj : input) {
result.add(processor.apply(obj));
}
return result;
}
public static void main(String[] args) {
List input = Arrays.asList(new String[] {"apple", "orange", "pear"});
List lengths = CollectionUtils.map(input, (String v) -> v.length());
List uppercases = CollectionUtils.map(input, (String v) -> v.toUpperCase());
}
}
自定义函数式接口及@FunctionalInterface注解
注解可以起到指示、约束作用,关于注解大家可以参考 Java 注解详解 (annotation)。Java 8 为函数式接口新提供了@FunctionalInterface
注解,当用注解的接口不是有效的函数式接口时,也就是接口中不只有一个抽象方法时,编译器会报编译错误。我们自定义一个函数式接口,并用@FunctionalInterface
进行注解
@FunctionalInterface
public interface MyFunction {
public void doSomething();
}
如果我们在上面接口中再加一个抽象方法:
@FunctionalInterface
public interface MyFunction {
public void doSomething();
public void doAnotherThing();
}
由于自定义接口中出现了两个抽象方法,不符合函数式接口定义,此时会抛出编译错误。我们再测一下上面这个自定义函数式接口:
public class TestMyFunction {
public static void execute(MyFunction worker) {
worker.doSomething();
}
public static void main(String [] args) {
//传统匿名类的方式调用
execute(new MyFunction() {
public void doSomething() {
System.out.println("Worker invoked using Anonymous class");
}
});
//通过Lambda表达式调用
execute( () -> System.out.println("Worker invoked using Lambda expression") );
}
}