为什么使用 Lambda 表达式
Lambda是一个匿名函数,我们可以把 Lambda 表达式理解为是一段可以传递的代码(将代码像数据一样进行传递)。使用它可以写出更简洁、更灵活的代码。作为一种更紧凑的代码风格,使Java的语言表达能力得到了提升
Lambda表达式使用举例
public class FieldTest {
@Test
public void test1() {
//提供Runnable接口的匿名实现类对象
Runnable r1=new Runnable() {
@Override
public void run() {
System.out.println("最可爱的人");
}
};
//注意这并不是多线程,因为也没有调用start方法
r1.run();//最可爱的人
System.out.println("***************使用Lambda表达式的方式进行重写***************");
Runnable r2=()-> System.out.println("最可爱的人");//Lambda表达式,->是Lambda操作符,或者叫箭头操作符
r2.run();//最可爱的人
}
@Test
public void test2(){
//提供Comparator的匿名实现类的对象
Comparator<Integer> com1=new Comparator<Integer>() {
@Override
public int compare(Integer o1, Integer o2) {
return Integer.compare(o1,o2);
}
};
int compare = com1.compare(12, 21);
System.out.println(compare);//-1
System.out.println("***************使用Lambda表达式");
//理解:compare中的参数类型已经由泛型决定了,所以不用写,Comparator接口只有一个抽象方法,所以也没有必要写方法名,反正只有这一个,所以能省的就省
Comparator<Integer> com2 = (o1,o2)->Integer.compare(o1,o2);//(o1,o2)->Integer.compare(o1,o2)就是Lambda表达式
int compare2=com2.compare(12,21);
System.out.println(compare2);//-1
}
}
Lambda表达式语法的使用:
举例:
(o1,o2)->Integer.compare(o1,o2)
格式:
->叫做Lambda操作符或者叫箭头操作符
它的左边称为Lambda形参列表,实际上就是接口中抽象方法的形参列表(只不过上面把类型也省略了,补上也行)
它的右边称为Lambda体,其实就是重写的抽象方法的方法体
Lambda表达式的使用(分为6中情况):
public class FieldTest {
@Test
public void test1(){
//语法格式一:无参,无返回值,Runnable就满足
Runnable r1=new Runnable() {//=的右边整体充当了Runnable接口实现类的对象,在下面的程序中就被替换为了()-> System.out.println("最可爱的人")
@Override
public void run() {
System.out.println("最可爱的人");
}
};
r1.run();//最可爱的人
System.out.println("***************使用Lambda表达式的方式进行重写***************");
//Lambda表达式的本质就是作为接口的实例,所以说还是作为对象出现
Runnable r2=()-> {
System.out.println("最可爱的人");
};//Lambda表达式,->是Lambda操作符,或者叫箭头操作符
r2.run();//最可爱的人
}
@Test
public void test2(){
//语法格式二:Lambda 需要一个参数,但是没有返回值
Consumer<String> com=new Consumer<String>() {//Consumer是接口
@Override
public void accept(String s) {
System.out.println(s);
}
};
com.accept("落叶堆积了好几层");//落叶堆积了好几层
System.out.println("***********Lambda表达式写法*************");
Consumer<String> con=(String s)-> {
System.out.println(s);
};
con.accept("落叶堆积了好几层");//落叶堆积了好几层
}
@Test
public void test3(){
//语法格式三:数据类型可以省略,因为可由编译器推断得出,称为“类型推断”
Consumer<String> con=(String s)-> {
System.out.println(s);
};
con.accept("落叶堆积了好几层");//落叶堆积了好几层
System.out.println("***************************");
//在上面的基础上进行优化
//String s的String可以省略,因为已经被泛型决定了,不可能是别的类型
Consumer<String> con1=(s)-> {
System.out.println(s);
};
con1.accept("落叶堆积了好几层");//落叶堆积了好几层
}
@Test
public void test4(){
//语法格式四:Lambda 若只需要一个参数时,参数的小括号可以省略
Consumer<String> con1=(s)-> {
System.out.println(s);
};
con1.accept("落叶堆积了好几层");//落叶堆积了好几层
System.out.println("***************************");
//在上面的基础上进行优化
Consumer<String> con2=s-> {
System.out.println(s);
};
con2.accept("落叶堆积了好几层");//落叶堆积了好几层
}
@Test
public void test5() {
//语法格式五:Lambda 需要两个或以上的参数(小括号不能省),多条执行语句,并且可以有返回值
Comparator<Integer> com1 = new Comparator<Integer>() {
@Override
public int compare(Integer o1, Integer o2) {
System.out.println(o1);
System.out.println(o2);
return o1.compareTo(o2);
}
};
System.out.println(com1.compare(12, 21));
System.out.println("***********Lambda表达式写法*************");
Comparator<Integer> com2 = (o1, o2) -> {
System.out.println(o1);
System.out.println(o2);
return o1.compareTo(o2);
};
System.out.println(com2.compare(12,21));
}
@Test
public void test6(){
//语法格式六:当 Lambda 体只有一条语句时,return 与大括号若有,都可以省略(是省略return,不是把return那一整句省略)
//比如如果只有一条执行语句,但不是return也是可以把大括号省略的
Comparator<Integer> com1 = (o1, o2) -> {
return o1.compareTo(o2);
};
System.out.println(com1.compare(12,21));
System.out.println("***********Lambda表达式写法*************");
Comparator<Integer> com2 = (o1, o2) -> o1.compareTo(o2);
System.out.println(com2.compare(12,21));
}
}
总结:
Lambda形参列表的的参数类型都可以省略,因为有类型推断
如果参数列表只有一个参数,其小括号也可以省略。
如果是没有参数或者是有一个及以上参数,不能省略小括号
Lambda体应该使用一对大括号进行包住,但特殊情况可以变化
如果Lambda体只有一条执行语句(也可以是return语句),则可以省略大括号,和return关键字
Lambda表达式本质上是接口的实例,要求接口必须是函数式接口