快速掌握Java中Lambda表达式的用法

时间:2021-04-03 18:57:51

Lambda表达式的作用:

  Lambda表达式的作用就是简化代码开发,让代码看起来更加简介。它是用来简化匿名内部类的。但是并不是所有的匿名内部类都能用Lambda表达式简化,Lambda表达式是有使用前提的。

 

 

Lambda表达式的使用前提:

  1、用Lambda表达式简化的这个匿名内部类必须是某一个接口的实现类,且这个实现的接口中有且只有一个抽象方法;因为只有这样,Lambda才能够根据接口中的上下文(代码)来推断出你简化的地方是什么内容,下面就来实际演示一下。(注:有且只有一个抽象方法的接口称为“函数式接口”)

  2、使用Lambda表达式的地方必须具有上下文推断,就是在使用Lambda表达式的地方知道能够通过代码上下文推断出所需要的信息,因为Lambda表达式简化匿名内部类是不会写出匿名内部类实现所实现的接口,所以需要在能够推测出接口的地方使用(有点抽象,可以通过下面的案例来理解)。

 

 

Lambda表达式的格式:

  Lambda表达式主要由三部分构成:()->{ }

    1、开头小括号开始 ( ) :小括号中填写的是重写接口中的方法时,方法的参数。若方法无参数,则小括号空着,若有多个参数,则参数间用逗号隔开(与普通Java方法相同)

    2、中间一个箭头 ->:可以理解为将参数传入到方法体中;

    3、末尾一个大括号{ }:大括号中的内容就是方法体,也就是方法的逻辑代码;

  当然,上面的结构是标准结构,但却不是绝对的,在某些情况下,部分内容可以省略,在下面的案例中将进行详细讲解。

 

 

Lambda使用案例:

  案例一:Lambda表达式简化匿名内部类,重写无参数无返回值方法

  首先,我们创建一个接口,用来测试Lambda表达式。通过Lambda表达式的使用前提可知,这个接口应当有且只有一个抽象方法。接口名字叫Test,方法叫display():

1 public interface Test {
2     
3     public abstract void display();
4 
5 }

  然后,我们创建一个类,这个类中有一个方法useDisplay(Test t),它唯一的参数就是上面定义的接口Test。我们知道,当要调用这个方法时,应当传入一个Test接口的实现类,方法代码如下:

1 public static void useDisplay(Test t) {
2     //方法的内容就是调用接口中的display方法
3     t.display();
4 }

  接下来,我们调用这个方法,分别采用匿名内部类以及Lambda表达式两种方式,以此来进行对比:

 1 //调用方法,接口参数使用匿名内部类当场实现
 2 useDisplay(
 3     new Test() {
 4         //重写接口中的方法,输出一句话
 5         @Override
 6         public void display() {
 7             System.out.println("Lambda表达式案例一");
 8         }
 9     }
10 );

  上面的代码调用了useDisplay(Test t)方法,并在括号中使用匿名内部类实现了Test接口以及重写了Test的方法,将匿名内部类作为参数传递。这是匿名内部类的方式,那下面看看Lambda表达式时怎么去写的:

1 useDisplay(
2     //使用Lambda实现
3     ()->{
4         System.out.println("Lambda表达式案例一");
5     }
6 );

  可以看到,代码明显精简了许多,我们来仔细看看究竟做了哪些改变:我们省略new,也省略了接口的名字,直接重写接口中的方法。为什么可以这么做呢,这就关系到我们之前所说的Lambda使用前提2:使用Lambda表达式的地方必须具有上下文推断;Lambda知道useDisplay的参数是Test接口,所有Lambda知道这里应该实现的接口就是Test,不可能是其他的了,因此可以省略接口名。接下来省略了方法的public、返回值类型、方法名,这些Lambda也可以推断出来,因为已经确定了是Test接口,那自然也知道了这个接口中的方法display()的信息,所以统统可以省略,这也是Lambda表达式使用前提1的原因。

  然而,上面的Lambda表达式只是标准格式,它还能够继续简化,上面的表达式简化后的如下:

1 useDisplay(  ()->System.out.println("Lambda表达式案例一") /*此处原本有分号*/ );

  可以看到,上面的Lambda表达式相对于之前,省略了大括号以及末尾的分号。只有在Lambda表达式的方法体,也就是大括号中只有一条语句时,才能够省略大括号,同时也要省略分号,两者要么一起省略,要么都不省略。

 

 

  案例二:Lambda表达式简化匿名内部类,重写有参有返回值方法

  同样是定义一个接口Calculator,里面只有一个抽象方法add(),计算两个整数的和:

1 public interface Calculator {
2     
3     public abstract int add(int a,int b);
4 
5 }

  与上一个案例类似的结构,在另一个类中定义一个方法invokeAdd,方法的内容就是调用接口中的add方法,得到两个整数的和并输出:

1 public static void invokeAdd(int a,int b,Calculator cal) {
2     System.out.println(cal.add(a, b));
3 }

  首先还是和第一个案例一样,我们使用匿名内部类的方式调用这个方法(这次我把代码写的紧凑一些,上一个案例代码都拉的很开,方便查看):

1 invokeAdd(10, 20, new Calculator() {
2     @Override
3     public int add(int a, int b) {
4         return a+b;
5     }
6 });

  上面代码很简单,传入两个要相加的整数10和20,以及一个匿名内部类对象实现,实现Calculator接口,重写接口中add方法,求两个整数的和。下面来看看Lambda表达式怎么写上面这个代码:

1 //使用Lambda表达式的方式代替匿名内部类实现
2 invokeAdd(10, 20, (int a,int b)->{
3     return a+b;
4 });

  还是来看看改变了哪些内容:首先,这是比较标准的写法,与案例一相同,省略了接口,直接重写接口中的方法,然后省略了方法的名字类型等,只留下一个大括号,里面包含方法的参数,后面跟着一个箭头,指向了方法体,而方法体则未改变。但是,这并不是最简略的写法,看下面这种更加简洁的写法:

1 //上Lambda表达式的简化
2 invokeAdd(10, 20, (a,b) -> a+b );

  可以看到,这时已经简化到只剩一句代码了,我们来仔细看看简化了哪些内容:首先,填写方法参数的小括号中,参数的类型省略了,还是和上面一样的原因,既然Lambda表达式已经知道了你实现的接口,就知道了接口中那唯一的方法,也就知道了方法中参数的类型,所以即使你省略,Lambda也能够根据上下文推测出来;其次,因为方法体中只有一句代码,所以可以省略大括号以及方法体中语句末尾的分号;除此之外,这是一个有返回值的方法,在方法体中只有一句代码的情况下,连return这个关键字都能省略。(注意:大括号,逗号,return关键字,要么一起省略,要么都不省略。

 

  最后再说一个可以简化Lambda表达式的情况,就是当接口中的方法只有一个参数时,连填写参数的小括号都可以省略,这里就不演示了。