1.使用lambda表达式的前提需要定义一个函数接口
-
函数接口:有且仅有一个抽象方法的接口
-
可以通过
@FunctionalInterface
注解去标记一个接口表示这是一个函数接口(该注解会检查接口是否只有一个抽象方法),没有这个注解也是可以的。
-
java.util.function 包下定义了很多函数接口
-
lambda表达式只能用在赋值,调用方法传参,类型转换的语句中。
2.lambda表达式用于描述如何实现一个函数接口的抽象方法
-
语法()->{}
左边是方法参数列表,右边是方法体
参数列表中的参数可以不指定类型,lambda会根据上下文推断参数类型 -
也可以通过双冒号引用现有方法来描述抽象方法的实现,如list.forEach(log::info)
3.lambda表达式求值时会产生对应函数接口的实例,但不会执行方法体。
- 这个实例是一个匿名内部类的对象,这个匿名内部类的class是在运行时生成的。
- 跟匿名类一样lambda方法体内部可以访问局部变量,不过局部变量会强制变为final类型(java 8 Effectively final ),就是不能对局部变量进行重新赋值。
- 分析lambda产生的实例可知,lambda方法体中的局部变量会变成匿名内部类的成员
public class TestLambda {
public static void main(String[] args) {
TestLambda t = new TestLambda();
t.test();
System.out.println("end");
}
public void test(){
List<String> list = new ArrayList<>();
Integer i = 0;
String ss = "ss";
//对lambda的求值会产生对应函数接口的实例
ITestLambda a = s -> {
//引用外部变量
list.add(s);
//引用外部变量
System.out.println("i=" + i);
//引用外部变量
System.out.println("ss=" + ss);
//引用外部变量
System.out.println("lambda list[0] :" + list.get(0));
return list.get(0);
};
a.test("e");
System.out.println("内部类:" + a.getClass());
System.out.println("构造器第1个参数:" + a.getClass().getDeclaredConstructors()[0].getParameters()[0].getType().getName());
System.out.println("构造器第2个参数:" + a.getClass().getDeclaredConstructors()[0].getParameters()[1].getType().getName());
System.out.println("构造器第3个参数:" + a.getClass().getDeclaredConstructors()[0].getParameters()[2].getType().getName());
System.out.println("test list[0] :" + list.get(0));
}
4.举例
- forEach方法在传参是用到lambda
public class TestLambda {
public static void main(String[] args) {
List<String> list = new ArrayList<>();
list.add("1");
list.add("2");
list.add("3");
list.forEach(n-> System.out.println(n+"="));
}
- forEach方法定义,入参是一个函数接口Consumer,后续调用该接口的accept方法。
- forEach方法入参Consumer
- lambda用在这里就是对Consumer接口 accept方法的实现,在求值时会创建该接口的实例作为forEach方法的参数传入,后续forEach调用传入的接口来完成forEach的具体逻辑。