目录
概述
Lambda表达式是Java8才新增的表达式。Lambda表达式本质是一个匿名方法。Lambda没有方法名称,但有参数列表、函数主体、返回类型,可能还有一个可以抛出的异常列表。使用Lambda表达式可以写出更简洁、更灵活的代码。
特点
- 匿名,因为它不像普通的方法那样有一个明确的名称。
- 函数,因为Lambda函数不像方法那样属于某个特定的类。但和方法一样,Lambda有参数列表、函数主体、返回类型,还可能有可以抛出的异常列表。
- 简洁,无需像匿名类那样写很多模板代码。
- 传递,Lambda表达式可以作为参数传递给方法或存储在变量中。便于并行
- 最大化减少空指针异常Optional
语法格式
Lambda表达式在Java 语言中引入了一个新的语法元素和操作符。这个操作符为 “->” , 该操作符被称 为 Lambda 操作符或剪头操作符。它将 Lambda 分为两个部分:
左侧: 指定了 Lambda 表达式需要的所有参数
右侧: 指定了 Lambda 体,即 Lambda 表达式要执行的功能。
(parameters) -> expression或(parameters) ->{ statements; }
可选类型声明:不需要声明参数类型,编译器可以统一识别参数值。
可选的参数圆括号:一个参数无需定义圆括号,但多个参数需要定义圆括号。
可选的大括号:如果主体包含了一个语句,就不需要使用大括号。
可选的返回关键字:如果主体只有一个表达式返回值则编译器会自动返回值,大括号需要指定明表达式返回了一个数值。
以下是Lambda表达式的语法格式:
- 语法格式一:无参,无返回值,Lambda 体只需一条语句。示例:Runnable runnable = () -> System.out.println("Lambda!");
- 语法格式二:Lambda 需要一个参数。示例:Consumer<String> consumer =(p)-> System.out.println(p);
- 语法格式三:Lambda 只需要一个参数时,参数的小括号可以省略。示例:Consumer<String> consumer = p -> System.out.println(p);
- 语法格式四:Lambda 需要两个参数,并且有返回值 。
示例:Comparator<Integer> com = (x, y) -> { System.out.println("lambda"); return Integer.compare(x, y); };
- 语法格式五:当 Lambda 体只有一条语句时,return 与大括号可以省略。示例:Comparator<Integer> comparator = (x, y) -> Integer.compare(x, y);
- Lambda 表达式的参数列表的数据类型可以省略不写,因为JVM编译器通过上下文推断出,数据类型,即“类型推断”。
示例:Comparator<Integer> comparator = (Integer x, Integer y) -> { //Integer 类型可以省略 System.out.println("lambda"); return Integer.compare(x, y); };
类型推断:Lambda 表达式中的参数类型都是由编译器推断 得出的。 Lambda 表达式中无需指定类型,程序依然可 以编译,这是因为 javac 根据程序的上下文,在后台 推断出了参数的类型。 Lambda 表达式的类型依赖于上 下文环境,是由编译器推断出来的。这就是所谓的 “类型推断”。
使用场景
函数式接口
只包含一个抽象方法的接口,称为函数式接口。Lambda表达式允许你直接以内联的形式为函数式接口的抽象方法提供实现,并把整个表达式作为函数式接口的实例(具体说来,是函数式接口一个具体实现的实例)。
示例:
new Thread(() -> {
System.out.print("test");
}).start();
Java 内置核心函数式接口 :
- Consumer<T> : 消费型接口 void accept(T t);
- Supplier<T> : 供给型接口 T get();
- Function<T, R> : 函数型接口 R apply(T t);
- Predicate<T> : 断言型接口 boolean test(T t);
方法引用
当要传递给Lambda体的操作,已经有实现的方法了,可以使用方法引用(实现抽象方法的参数列表,必须与方法引用方法的参数列表保持一致)。
方法引用:使用操作符 “::” 将方法名和对象或类的名字分隔开来。
对象::实例方法示例:
PrintStream ps = System.out;
Consumer<String> con = ps::println;
类::静态方法示例:
Consumer<String> consumer = System.out::println;
需要注意:
- 方法引用所引用的方法的参数列表与返回值类型,需要与函数式接口中抽象方法的参数列表和返回值类型保持一致!
- 若Lambda 的参数列表的第一个参数,是实例方法的调用者,第二个参数(或无参)是实例方法的参数时,格式: ClassName::MethodName
构造器引用
构造器的参数列表,需要与函数式接口中参数列表保持一致!
格式: ClassName::new
与函数式接口相结合,自动与函数式接口中方法兼容。可以把构造器引用赋值给定义的方法,与构造器参数列表要与接口中抽象方法的参数列表一致。
示例:
Function<Integer,MyClass> myClass = MyClass::new;
常见写法
- 替代匿名内部类
new Thread(()-> System.out.print("test lambda")).start();
- 使用stream()中的forEach()/filter()/map()等方法 对集合进行处理
int[] nums = {1,2,3,4,5}; List<Integer> list = Arrays.stream(nums).boxed().collect(Collectors.toList()); list.stream().filter(e -> e > 2) .forEach(e -> System.out.println(e.toString())); list.stream().map(x -> x + x*5).forEach(System.out::println);