今天在面试的时候被面试官问到这个问题,之前也了解过,但是将的的时候些许吃力,所以整理下来。
Lamada是一个匿名函数,本质上是一个“语法糖”。
可以把 Lambda表达式 理解为是一段可以传递的代码 (将代码像数据一样进行传递)。
可以写出更简洁、更灵活的代码。作为一种更紧凑的代码风格,使Java的语言表达能力得到了提升。
Lambda 表达式的基础语法 :
Java8 中引入了一个新的操作符 "--------->" 该操作符称为箭头操作符或 Lambda 操作符,箭头操作符将 Lambda 表达式拆分成两部分 :
左侧 : Lambda 表达式的参数列表
右侧 : Lambda 表达式中所需执行的功能, 即 Lambda 体
语法格式一 : 无参数,无返回值
() -> System.out.println("Hello Lambda!");
语法格式二 : 有一个参数,并且无返回值
(x) -> System.out.println(x)
语法格式三 : 若只有一个参数,小括号可以省略不写
x -> System.out.println(x)
Consumer<String> con = (x) -> System. out .println(x);
con.accept( "啦啦啦,我是卖报的小行家" );
语法格式四 : 有两个以上的参数,有返回值,并且 Lambda 体中有多条语句
Comparator <Integer> com = (x, y) -> {
System.out.println("函数式接口");
return Integer.compare(x, y);
};
语法格式五 : 若 Lambda 体中只有一条语句, return 和 大括号都可以省略不写
Comparator <Integer> com = (x, y) -> Integer.compare(x, y);
语法格式六 : Lambda 表达式的参数列表的数据类型可以省略不写,因为JVM编译器通过上下文推断出,数据类型,即“类型推断”
(Integer x, Integer y) -> Integer.compare(x, y);
注 : Lambda 表达式中的参数类型都是由编译器推断得出的。
Lambda 表达式中无需指定类型,程序依然可以编译,这是因为 javac 根据程序的上下文,在后台推断出了参数的类型。 Lambda 表达式的类型依赖于上下文环境,是由编译器推断出来的。这就是所谓的 “类型推断”。
上联 : 左右遇一括号省
下联 : 左侧推断类型省
横批 : 能省则省
Lambda 表达式需要 “函数式接口” 的支持
函数式接口 : 接口中只有一个抽象方法的接口,称为函数式接口,可以通过 Lambda 表达式来创建该接口的对象 (若 Lambda表达式抛出一个受检异常,那么该异常需要在目标接口的抽象方法上进行声明) 可以使用注解 @FunctionalInterface 修饰可以检查是否是函数式接口,同时 javadoc 也会包含一条声明,说明这个接口是一个函数式接口。
@FunctionalInterface
public interface MyFun {
Integer getValue(Integer num);
}
函数式接口中使用泛型 :
@FunctionalInterface
public interface MyFun< T > {
T getValue( T t);
}
从匿名类到 Lambda 的转换
// 原来的匿名内部类作为参数传递
Comparator<Integer> comparator = new Comparator<Integer>() {
@Override
public int compare(Integer o1, Integer o2) {
return Integer. compare (o1, o2);
}
};
TreeSet<Integer> treeSet = new TreeSet<>(comparator);
// Lambda 表达式作为参数传递
Comparator<Integer> comparator = (o1, o2) -> Integer. compare (o1, o2);
TreeSet<Integer> treeSet = new TreeSet<>(comparator);
作为参数传递 Lambda 表达式 : 为了将 Lambda 表达式作为参数传递,接收Lambda 表达式的参数类型必须是与该 Lambda 表达式兼容的函数式接口的类型。