Java Lambda 表达式介绍
Posted Heaven-Wang
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了Java Lambda 表达式介绍相关的知识,希望对你有一定的参考价值。
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") );
以上是关于Java Lambda 表达式介绍的主要内容,如果未能解决你的问题,请参考以下文章
Kotlin函数 ⑤ ( 匿名函数变量类型推断 | 匿名函数参数类型自动推断 | 匿名函数又称为 Lambda 表达式 )