借助一段代码进行lambda表达式的分析:

public class Main {
    public static void main(String[] args) {
        MathOper addOper=new MathOper() {
            @Override
            public int oper(int a, int b) {
                return a+b;
            }
        };
        MathOper subOper=new MathOper() {
            @Override
            public int oper(int a, int b) {
                return a-b;
            }
        };
        oper(3,4,addOper);
        oper(3,4,subOper);
    }
    static void oper(int a,int b,MathOper mathOper){
        System.out.println(mathOper.oper(a,b));
    }
}
interface MathOper{
    int oper(int a,int b);
}
View Code

上面的代码代码中,MathOper是一个很简单的接口,里面只包含了一个抽象方法,为了方便起见,我没有创建它的实现类,而是在使用的时候,直接利用MathOper接口创建响应的匿名类对象,并在匿名类中实现oper方法,然后在static void oper方法中调用相应的匿名类对象去进行相应的计算,上面的操作方法虽然可以实现相应的功能,但是代码结构不是那么紧凑,是不是还可以更简单一点,有的:

public class Main {
    public static void main(String[] args) {
        MathOper addOper = (a, b) -> a + b;
        MathOper subOper = (a, b) -> a - b;
        oper(3, 4, addOper);
        oper(3, 4, subOper);
    }
    static void oper(int a, int b, MathOper mathOper) {
        System.out.println(mathOper.oper(a, b));
    }
}

interface MathOper {
    int oper(int a, int b);
}
View Code

 上面的代码中,我没有创建匿名类对象,只是将抽象函数的”操作结构“负值给了addOper,subOper变量,然后再在static void oper方法使用这些变量,上面的代码结构能不能再进一步优化呢?既然static void oper方法中已经有了mathOper变量声明,我们能不能把addOper,

subOper变量也一并省了,直接传递”操作结构“呢?答案是可以的,如下:

public class Main {
    public static void main(String[] args) {
        oper(3, 4, (a, b) -> a + b);
        oper(3, 4, (a, b) -> a - b);        
    }
    static void oper(int a, int b, MathOper mathOper) {
        System.out.println(mathOper.oper(a, b));
    }
}

interface MathOper {
    int oper(int a, int b);
}
View Code

 上面的代码更加紧凑了,没有声明多余的变量,直接将函数体传递给了oper方法,由oper方法去解析执行相应的操作,上面的这种紧凑的语法表达形式就是Lambda表达式,Lambda表达式的语法结构组成有:

1、由括号(”()“)包裹起来的并且被分号(",")隔开的参数,如上面的(a, b)

2、水平右指的箭头("->")

3、代码体,可由大括号("{}")包裹起来的代码块,也就是具体的功能实现语句,如果是简单的一条执行语句,则不需要由大括号包裹,如上面的a+b,再举一个对数组对象进行排序的例子,如下:

public class Main {
    public static void main(String[] args) {
        Person[] pers={new Person(12,"h1"),
                new Person(23,"h2"),
                new Person(14,"h3"),
                new Person(2,"h4")};
        Arrays.sort(pers,(b1,b2)->{
            return b1.getAge()-b2.getAge();
        });
        Arrays.stream(pers).forEach(obj->System.out.print(obj+" "));
    }
}
@Data
@ToString
class Person{
    int age;
    String name;
    public Person(int age,String name){
        this.age=age;
        this.name=name;
    }
}
View Code

上面的Arrays.sort函数的参数有两个:T[] a, Comparator<? super T> c;其中Comparator接口有一个注解:@FunctionalInterface,这个注解表明该接口是一个函数型接口,函数型的接口只能有一个抽象函数,该抽象接口的实现可用lambda表达式进行表示;但是进入Comparator接口可以发现,其方法并非只有一个,但是其他方法有一个特点,就是被default关键字修饰了,这里引出Java8的另一个新特性:Default Method

默认方法是接口中定义的一个具有方法实现体的方法,我们知道,接口中的抽象方法必须被其实现类进行重写,但是接口中的默认方法可由实现类直接继承,然后由其实现对象直接拿来用;如下图所示:

public class Main {
    public static void main(String[] args) {
        MathOper mathOper=new TribleMathOper();
        //实现类对象直接调用默认方法
        System.out.println(mathOper.myDefault(23));
        //lambda方法实现抽象方法的实现体
        MathOper lambdaOper=(a,b)->a*b;
        System.out.println(lambdaOper.oper(2,4));
    }
}
@FunctionalInterface
interface MathOper {
    int oper(int a, int b);
    default int myDefault(int a){
        return 2*a;
    }
}
class TribleMathOper implements MathOper{
    @Override
    public int oper(int a, int b) {
        return a+b;
    }
}
View Code

上面的实现类对象可以直接调用继承过来的默认方法,有了默认方法的功能,接口越来越像抽象类了

以上是lambda表达式使用总结,道阻且长,未来可期......

 

posted on 2022-01-26 19:21  Judy518  阅读(35)  评论(0编辑  收藏  举报