10 分钟快速搞懂 Lambda 表达式

Lambda简介

Lambda表达式是Java8引入的一个重要特性,相当于一个语法糖。

语法糖(Syntactic sugar)是指在编程语言中引入的一种语法,它可以使代码更易读、更简洁,但并没有引入新的功能或改变语言的底层机制。语法糖并不会改变语言的语义,只是提供了一种更方便的编写方式。

  • Lambda表达式可以被视为匿名函数
  • 允许在需要函数的地方以更简洁的方式定义功能

使用条件:只要是函数式接口就可以用Lambda表达式简化

函数式接口:接口中有且只有一个未实现的方法,这个接口就叫函数式接口

如果接口中有超过一个未实现方法,则不是函数式接口,不能用Lambda表达式 如果接口中有一个未实现方法,有一个默认实现方法,则是函数式接口,可以用Lambda表达式

如:

// 接口中有超过一个未实现方法,不是函数式接口
interface MyInterface {
    int sum(int a, int b);
    int min(int a, int b);
}

// 接口中只有一个未实现的方法,是函数式接口
interface MyCase{
    int hello();
    default int hello(int a){return a;} //默认实现
}

可以用jdk中提供的检查注解@FunctionalInterface来检查该接口是否为函数式接口

正常情况:

异常情况:

Lambda表达式与函数式接口

举个例子:

有一个接口MyInterface,里面有一个方法sum()。

interface MyInterface {
    int sum(int a, int b);
}

如果想要实现这个接口,我们可以使用两种方法。

1.自己写实现类

class MyInterfaceImpl implements MyInterface {
    @Override
    public int sum(int a, int b) {
        return a + b;
    }
}

然后调用

public class Lambda {
    public static void main(String[] args) {
        MyInterface myInterface = new MyInterfaceImpl();
        int result = myInterface.sum(1, 2);
        System.out.println(result);
    }
}

2.创建匿名实现类如果每个接口都要写实现类的话,总觉得会有点麻烦。而且,如果在以后的业务中,要实现的方法不是两数之和,而是两数的平方和的话,那我岂不是要再写一个实现类?为了解决这个问题,我们可以采用匿名实现类,动态的去实现接口。

public class Lambda {
    public static void main(String[] args) {

        // 1. 自己创建实现类对象
        MyInterface myInterface = new MyInterfaceImpl();
        int result = myInterface.sum(1, 2);
        System.out.println("我是 自己创建的实现类对象 " + result);

        // 2. 创建匿名实现类
        MyInterface myInterface1 = new MyInterface() {
            @Override
            public int sum(int a, int b) {
                return a*a + b*b;
            }
        };
        int result1 = myInterface1.sum(1, 2);
        System.out.println("我是 匿名实现类 " + result1);
    }
}

运行结果:

可以看到,每次创建匿名实现类的时候,有很多格式上的东西是每次都要写的,这样就很冗余。比如下面我选中的这部分:

这部分在上面的接口interface MyInterface中就已经定死了。就算不写new MyInterface(),从创建匿名实现类的前面部分MyInterface myInterface1,也可以看出实现的就是MyInterface接口。

与前面相比,lambda表达式只保留动态的东西,把写死的东西去掉。 用lambda表达式的方式实现接口:

// 3. Lambda表达式实现接口  参数列表 + 箭头 + 方法体
MyInterface myInterface2 = (int a, int b) -> {
    return a * a + b * b;
};

这是lambda表达式的完整写法,然而我们可以看到,入参的类型在接口中也是定好的。这就说明还有更简化的写法:

1.参数类型可以不写,只写参数名,参数变量名随意定义

MyInterface myInterface3 = (x, y) -> {
    return  x * x + y * y;
};

参数名不一定是接口中定义的(a,b),也可以定义为其他的名字,比如(x,y)

2.参数部分在没有入参的时候,最少可以只有一个(),但是不能不写括号!!!!或者在只有一个入参的时候,只有一个参数名

interface MyCase{
    int hello();
}

public class Lambda {
    public static void main(String[] args) {
        MyCase myCase = () -> {
            return 1;
        };
    }
}
---------------------------------------------------
interface MyCase1{
    int hello(int a);
}

public class Lambda {
    public static void main(String[] args) {
        MyCase1 myCase1 = a -> {
            return a + 1;
        };
    }
}

3.方法体只有一句话的时候,{}和return可以省略

interface MyCase1{
    int hello(int a);
}

public class Lambda {
    public static void main(String[] args) {
        MyCase1 myCase11  = a ->  a + 2;
        // 调用方法
        System.out.println(myCase11.hello(1));
    }
}

Lambda表达式使用

未来使用函数时接口会比较频繁,当调用某个方法传入参数,这个参数实例是一个接口对象,且只定义了一个方法,就可以直接用Lambda简化写法

 

比较器的使用

当我们使用比较器的时候,可以使用Lambda表达式简化写法

public class Lambda {
    public static void main(String[] args) {
        ArrayList<String> names = new ArrayList<String>();
        names.add("Pidanxia");
        names.add("Lucy");
        names.add("Bob");
        names.add("Tom");

        // 普通写法
        Collections.sort(names,new Comparator<String>() {
            @Override
            public int compare(String o1, String o2) {
                return o1.compareTo(o2);
            }
        });

        // Lambda写法
        Collections.sort(names,(o1,o2)->o1.compareTo(o2));
    }
}

可以看到Lambda表达式与普通写法相比,简洁了非常多。 除了Lambda表达式之外,还可以用方法引用来简化写法。如:

Collections.sort(names, String::compareTo);

类::方法:表示引用类中的实例方法。

比如例子中的语句,就是说,以names为入参,调用String类中的compareTo方法

线程的使用

// 普通写法
new Thread(){
    @Override
    public void run() {
        System.out.println("Hello");
    }
}.start();
// Lambda写法
new Thread(()->System.out.println("Hello")).start();
posted @ 2024-07-18 14:13  柒墨轩  阅读(227)  评论(0)    收藏  举报