目录
1. 什么是lambda表达式?
lambda表达式是Java 8及其后续版本引入的一种语言特性,它是一个能接受输入,处理后得到返回结果的代码段。例如:(x,y)->{Math.pow(x,2)+Math.pow(y,2)},就是一个输入为x和y,输出为它们的平方和的代码段。
lambda表达式强调的是“给定什么,处理后得到什么”这一处理过程本身,不关心这个过程是由哪个对象完成的,它借鉴了函数式编程的简单易用的哲学思想。
2. 为什么要引入lambda表达式?
在一些应用开发场景中,我们需要的往往仅仅是一段可执行的代码,至于这段代码由哪个对象来完成,根本不重要,也用不着给这段代码起一个方法名或函数名。例如,在使用Thread类的带参构造方法Thread(Runnable runnable)创建一个Thread对象时,我们仅仅需要的是一段在新创建的线程对象中执行的代码段,至于这个代码段是由哪个Runnable对象负责的、它是什么名称,都不重要。此时,如何处理这一情况?
传统面向对象方法的实现流程是,创建一个实现Runnable接口的匿名类的对象,代码如下:
Thread thread2=new Thread(new Runnable() { //匿名Runnable对象
@Override
public void run() {
... //要执行的代码段
}
});
上述代码最关键的部分是run方法中要执行的代码段,至于这个匿名类的对象,run方法名都只是形式上的,没有实际作用。在这种场景下,可以用lambda表达式简化上述代码:
Thread thread2=new Thead(()->{... //要执行的代码段});
可见,lambda表达式,省掉了面向对象的条条框框,只关心处理过程,不关心该过程的负责对象及过程名称。它遵循极简主义,可极大地简化代码。它特别适用于需要一段“临时需要、用后即扔”的代码的场景。
3. 什么情况下使用lambda表达式?
lambda表达式是为了在不需要对象、不需要方法名称的情形下代替匿名对象而存在的。具体而言,只要满足下列两个条件的任意一个,即可用lambda表达式:
(1)某方法中的某个参数的类型为接口,且该接口仅包含1个抽象方法(这种接口称为函数式接口);
(2)某个局部变量的类型为接口,且该接口仅包含1个抽象方法。
在上述情况下,可以创建一个lambda表达式,它相当于一个匿名对象(记住:只是相当于,但不等价于,因为匿名类在编译时,会产生与匿名类相对应的class文件,而lambda表达式不会)。
-
4. 如何使用lambda表达式?
4.1 lambda表达式标准写法
(参数列表)->{
代码块
}
示例1:不带参数、无返回值的lambda表达式
()->{
System.out.println("yes");
}
示例2:不带参数、有返回值的lambda表达式
()->{
int a=3;
int b=4;
return a+b;
}
示例3:带参数、无返回值的lambda表达式
(double a,double b)->{
System.out.println("a+b"+(a+b));
}
示例4:带参数、有返回值的lambda表达式
(double a,double b)->{
return a+b;
}
4.2 lambda表达式的简化
lambda表达式的书写遵循“能够根据上下文推断出的部分,就可以省略“”的原则。下面列出了三种可以简化lambda表达式的条件及简化方案:
(1)带参数的lambda表达式中的参数类型可以省略。例如,上面的示例3可以简写为:
(a, b)->{
System.out.println("a+b"+(a+b));
}
(2)仅带一个参数,则参数左右的小括号可以省略。例如:
a->{
b=a+5;
return b;
}
(3)如果代码块只包含一条语句,可省略代码块两边的大括号、return关键字(如果有的话)及这条语句后面的分号。例如:
(a,b)->a+b //返回a,b之和
4.3 lambda表达式使用示例
假设有一个GUI程序,主界面上包含2个文本框(txtA、txtB),一个按钮(btnComputing),希望在txtA中输入一个整数a后,单击该按钮时,执行一段代码,求小于a的所有完数,输出到txtB中。由于这个计算过程太耗时,为避免其阻塞主窗口线程,将其放到一个线程中执行。这时候,可以利用lambda表达式,参考代码如下:
btnComputing.addActionListener(
e->{
try {
btnComputing.setEnabled(false);
txtB.setText("");
StringBuilder sb=new StringBuilder();
int a=Integer.valueOf(txtA.getText());
Runnable r=()->{
for(int k=2;k<a;k++) {
if(PerfectNumber.isYes(k)) {
sb.append(" "+k);
synchronized(this) {
txtB.setText(sb.toString());
}
}
}
btnComputing.setEnabled(true);
btnComputing.setText("开始计算");
};
btnComputing.setText("计算中...");
new Thread(r).start();
} catch (NumberFormatException e1) {
JOptionPane.showMessageDialog(null, "请输入一个整数!");
btnComputing.setEnabled(true);
txtA.requestFocus();
}
}
);
注:这里的addActionListener里面的参数就是一个lambda表达式,而r也是一个lambda表达式对象。