Java真的不难(二十四)Lambda表达式

Lambda表达式:

1、简介

首先Lambda表达式是属于Java8的 一个新特性,提供Java编程中对于函数式编程的支持,有助于代码的简洁,可以取代大半部分的匿名函数,尤其对于集合的遍历和集合的操作,极大的简化了代码。

Lambda表达式的主体:
在这里插入图片描述

函数式接口:

注意: Lambda表达式一定要配合函数式接口一起使用,所谓函数式接口,就是接口中只有一个抽象方法的接口就是函数式接口,我们可以自定义,JDK也内置了大量的函数式接口。

1、@FunctionalInterface注解修饰了接口,那这个接口就是函数式接口,只能有一个方法,下面就是一个函数式接口:

@FunctionalInterface
public interface MyInteface {

    void eat();

}

2、 如果不加@FunctionalInterface**注解,你在接口里面只写一个抽象方法也可以认为是函数式接口:

public interface MyInteface {
    void eat();
}

这样也是可以的。

3、 函数式接口只有一种情况不只有抽象方法,那就是可以继承Object类的方法:

@FunctionalInterface
public interface MyInteface3 {

    void eat();

    @Override
    String toString();
    
    @Override
    int hashCode();
}

2、Lambda表达式的使用:

1、在普通方法内的使用

Student类:

@FunctionalInterface
public interface Student {

    void eat();

}

测试类:

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

        Student stu = new Student() {

            //普通方法,重写并使用
            @Override
            public void eat() {
                System.out.println("我是学生");
            }
        };
        stu.eat();


        //lambda表达式写法:
        //参数1:重写了Student接口中唯一的那个无参数的eat抽象方法做了具体的实现,所以重写不 需要署名
        //参数2:-> 表达式 固定的
        //参数3:{具体的实现} 对Student接口中唯一的eat方法做了具体的实现
            Student stu2 = () -> {
                System.out.println("学生吃饭");
            };
        stu2.eat();
        
    }
}

输出:

我是学生
学生吃饭

2、带参方法的使用

Student类:

@FunctionalInterface
public interface Student {

    void eat(String food);
}

测试类:

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

        //lambda重写Student接口唯一的有参方法:
        Student stu2 = (foodName)->{
            System.out.println("学生在吃"+foodName);
        };

        stu2.eat("肉");
    }
}

//输出:学生在吃肉

3、Lambda表达式实现多线程

之前在多线程(1)的那篇文章内有介绍了创建多线程的方法,这里就使用lambda来创建线程:

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

        Thread t = new Thread(() -> {
            System.out.println("这个线程是由lambda来创建的");
        });

        t.start();

    }
}

4、Lambda表达式操作运算

我们使用lambda来操作运算可以少很多代码:

函数式接口:

@FunctionalInterface
public interface Calculator<T> {

    T operation(T v1,T v2);

}

测试类:

public class Test {

    //计算方法
    public static Integer operator(Integer v1,Integer v2,Calculator<Integer> calculator){
        return calculator.operation(v1, v2);
    }

    public static void main(String[] args) {


        //使用lambda表达式:
        //这里的意思就是传入两个参数,返回运行后的值
        int add = Test.operator(5,10,(x,y)->{
            return x+y;
        });


        //简写:可以少写很多代码,比上面更简介了
        int num1 = Test.operator(5,10,(x,y)->x+y);
        int num2 = Test.operator(10,5,(x,y)->x-y);

        System.out.println(add);
        System.out.println(num1);
        System.out.println(num2);

    }
}

输出:

15 、15 、5

5、Lambda表达式方法引用

有时候我们不是必须要要重写接口的方法来做具体的实现,我们如果有存在的方法能来实现,也可以通过方法 引用的方式来引用已经存在的方法做接口中方法具体的实现,这样的好处就是代码复用,比如下面这样:

函数式接口:

public interface ResultOneParam {

    int method(int a);
}

测试类:

public class Test {

    public int addTo(int a){
        return  a+10;
    }

    public  static int addTo2(int a){
        return  a+10;
    }

    public static void main(String[] args) {
        //lambda重写了method方法
        ResultOneParam lambda1=(a)->a+10;

        //方法引用:就是在Test里面的addTo2方法用来替代method被重写的方法
        ResultOneParam lambda2= Test::addTo2;
        int result1= lambda2.method(9);
        System.out.println(result1);

        //方法引用 ::引用现成的方法来替代方法重写,这样可以方法重用
        Test test=new Test();
        ResultOneParam lambda3=test::addTo;
        int result2= lambda3.method(9);
        System.out.println(result1);

    }
}

6、Lambda表达式对集合的使用

当然Lambda对集合的操作也是很方便的,可以少些很多代码:

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

        List<Integer> list = Arrays.asList(5,1,3,4,5,0,9,7,0,1,5);

        //lambda表达式遍历集合,重写了Consumer接口的方法
        list.forEach((element)->{
            System.out.println(element);
        });

        //简写:
        list.forEach(element-> System.out.println(element));

        //lambda表达式方法引用,用于遍历输出list集合:
        list.forEach(System.out::print);

        //输出list的偶数:
        list.forEach(element->{
            if(element%2==0){
                System.out.println(element);
            }

        });
    }
}

7、总结

好啦 以上就是Lambda表达式的一些基本用法,这也是Java8的一个新特性,至于原理以及其他方法,大家可以参考其他的技术博文,Lambda的出现极大的简化了代码,如果能熟练的运用lambda到实际业务中去,对于代码整体的整洁度有很大的提升!
在这里插入图片描述

Java Lambda 表达式Java 8 引入以来,为函数式编程提供了语法上的便利,但其在实际使用中也暴露出一些缺点。 ### 可读性下降 Lambda 表达式虽然减少了代码行数,但在某些情况下会降低代码的可读性。特别是当表达式体较为复杂时,嵌套的 Lambda 表达式会使逻辑难以理解。对于不熟悉函数式编程风格的开发者来说,理解和维护此类代码的成本较高[^1]。 ### 调试困难 由于 Lambda 表达式没有名称,调试时很难通过堆栈跟踪识别问题所在。这使得调试包含多个 Lambda 表达式的程序变得更为困难。此外,匿名性也增加了日志记录和异常追踪的难度。 ### 性能开销 尽管 Lambda 表达式在大多数场景下性能尚可接受,但它们可能会引入额外的运行时开销。例如,每次调用 Lambda 表达式都可能创建一个新的对象实例,这在频繁调用的情况下可能导致垃圾回收压力增加。根据一些基准测试结果,在特定高频率调用场景下,Lambda 的性能表现不如传统的匿名内部类。 ### 替代方案 #### 使用传统匿名内部类 在需要更高可读性和调试能力的场景下,可以继续使用传统的匿名内部类。虽然这种方式相比 Lambda 更加冗长,但它提供了明确的类型信息和方法名,有助于提高代码的清晰度[^1]。 ```java // 使用匿名内部类实现 Runnable 接口 new Thread(new Runnable() { @Override public void run() { System.out.println("Hello from thread"); } }).start(); ``` #### 使用方法引用与函数式接口组合 如果希望保留函数式编程的优势同时提升可读性,可以结合使用方法引用与定义良好的函数式接口。这样既能利用简洁的语法,又能通过命名的方法增强语义[^3]。 ```java // 定义一个函数式接口 @FunctionalInterface interface Operation { int apply(int a, int b); } // 使用方法引用 Operation add = Integer::sum; System.out.println(add.apply(5, 3)); // 输出 8 ``` #### 采用其他语言特性或工具库 为了进一步提升开发效率和代码质量,可以考虑采用如 Vavr(原 Javaslang)这样的库来引入更丰富的函数式数据结构和模式。或者探索 Kotlin 等现代 JVM 语言,它内置了对 lambda 和高阶函数的支持,并且语法更加简洁明了[^1]。 #### 使用 Stream API 的替代方式处理集合操作 虽然 Lambda 经常与 Stream API 一起使用以简化集合的操作,但在某些情况下直接使用传统的循环结构或者其他集合处理库(比如 Guava)也许会带来更好的性能以及更直观的代码结构[^3]。 通过以上替代方案,可以在一定程度上缓解 Lambda 表达式的局限性,并根据不同项目需求选择最合适的技术路径。
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值