Lambda表达式是可传递的代码块,可以执行一次到多次。
先来看个例子:
public class Test {
public static void main(String[] args) {
String[] strings = {"c", "a", "b"};
Arrays.sort(strings, (e1,e2)->e1.compareTo(e2));
}
}
(e1,e2)->e1.compareTo(e2) 就是一个可执行的代码块,它不是理解执行的,在完成排序之前,会一直调用这块代码。lambda表达式就是将代码块传递给对象,在将来某一个时间调用。
lambda 表达式的语法格式:
可选类型声明:不需要声明参数类型,编译器可以统一识别参数值。
(String x)->System.out.println(x);
(x)->System.out.println(x);
可选的参数圆括号:一个参数无需定义圆括号,但多个参数需要定义圆括号。
(x)->System.out.println(x);
x->System.out.println(x);
可选的大括号:如果主体包含了一个语句,就不需要使用大括号。
(x,y)->x-y;
(x,y)->{ return x-y;};
可选的返回关键字:如果主体只有一个表达式返回值则编译器会自动返回值,大括号需要指定明表达式返回了一个数值。
java中已经有很多封装了代码块的接口,lambda表达式与这些接口都是兼容的。对于只有一个抽象方法的接口,当需要这种接口的对象时,就可以提供一个lambda表达式。这种接口称为函数式接口。
举例:Runnable接口,只有一个run()的抽象方法。
@FunctionalInterface
public interface Runnable {
}
@FunctionalInterface 用来标记函数式编程接口
lambda表达式怎么转换成函数式接口的,还是以Runnable为例:
Thread thread = new Thread(new Runnable() {
@Override
public void run() {
System.out.println("hello");
}
});
thread.start();
Thread实例化,需要一个Runnable对象,这里面就是lambda表达式,这样看不明显,我们转下:
Thread thread = new Thread(() -> System.out.println("hello"));
thread.start();
这样就一目了然。
方法引用:
我们用Lambda表达式来实现匿名方法。但有些情况下,我们用Lambda表达式仅仅是调用一些已经存在的方
法,除了调用动作外,没有其他任何多余的动作,在这种情况下,我们倾向于通过方法名来调用它,而Lambda表达式可以帮助我们实现这一要求,它使得Lambda在调用那些已经拥有方法名的方法的代码更简洁、更容易理解。方法引用可以理解为Lambda表达式的另外一种表现形式。
方法引用 : ClassName :: staticMethodName
String[] strings = {"c", "a", "b"};
//lambda表达式
Arrays.sort(strings, (e1,e2)->{return e1.compareTo(e2);});
//等价于方法引用
Arrays.sort(strings, String::compareTo);
构造器引用 : ClassName :: new
//承接上面的代码
Stream<String> stream = Arrays.stream(strings).map(String::new);
变量的作用域:
lambda表达式只能引用final 标记的全局变量,说明只能引用,不能修改。
lambda表达式可以引用外层的局部变量,但是不能修改,说明隐性的加上final属性。
int num = 3;
Thread thread = new Thread(new Runnable() {
@Override
public void run() {
num++; //此处修改变量,编译器无法通过
System.out.println(num);
}
});
thread.start();