【精通函数式编程】(三)Lambda表达式原理与函数式接口精讲

简介: 本文讲解lambda表达式语法、语义,讲解函数式接口是什么,作用是什么,Lambda表达式的编译原理,以及Java8及高版本的函数式接口Consumer、Function 、Predicate,将会在工作中大量用到

image.png

前言:

📫 作者简介:小明 java 问道之路,专注于研究计算机底层,就职于金融公司后端高级工程师,擅长交易领域的高安全/可用/并发/性能的设计和架构📫

🏆 Java 领域优质创作者、阿里云专家博主、华为云享专家🏆

🔥 如果此文还不错的话,还请👍关注点赞收藏三连支持👍一下博主哦

本文导读

本文先通过语法讲解-》引出函数式接口-》Lambda原理解析-》函数式接口精讲,逐步深入

一、Lambda表达式语法精讲lambda是什么?

我们通过前面两篇文章,了解lambda表达式可以理解为一种简洁可传递的匿名函数,没有名称只有->,有参数列表函数的大括号{}(函数主体),还可以有返回值,可传递理解为可以作为参数传递给方法或存储在变量中。

如图是使用lambda表达式,排序代码的语法解析

image.png

Lambda表达式是不能直接使用 System.out.println 打印的 ,可以看到报了 “参数不匹配; *** 不是函数接口”,那什么是函数式接口?

二、函数式接口

image.png

函数式接口简单来说就是,只有一个抽象(abstract)方法的接口( interface),注如果我们只有一个抽象接口,但是有多个默认实现的方法,拿还是函数式接口

用函数式接口我们能做什么?Lambda表达式 说是类似匿名内部类,所以提供的这个抽象方法是作为函数式接口的实例的,可以理解为,Lambda表达式是 函数式接口的一个实现。

run0(() -> {
        System.out.println("执行run()内部逻辑");
    });
    // 由于Runnable中只定义了一个 抽象的 run() 方法,所以run方法就是一个函数式接口
    void run0(Runnable r) {
        r.run();
    }

这段代码很好的诠释了,什么是函数式接口,函数式接口怎么用lambda实现。

三、Lambda表达式原理解析

上面代码经过编译后, .class中共有三个方法,一个默认构造方法,一个main方法,还多了一个我们没有定义过的方法: lambdamain0()显然,这个方法就是lambda表达式内逻辑操作方法,它是一个static的方法。那么,这个lambdamain0()方法是怎么被调用的呢?

image.png

我们通过对java命令指定选项查看底层详细的编译过程:Java在编译时,首先,在实现类的内部,将Lambda表达式,抽取出来作为一个static方法lambdamain0();,然后对Lambda.class(final类)做了默认实现 $Lambda1.class,接下来,执行实现类的main()方法时,就会对lambda表达式利用实现类中的print()方法运行,最后将结果返回并打印。这就是Lambda表达式的底层运行逻辑。如图是上述代码的 .class文件

四、函数式接口详解

就Runnable来说,JavaAPI中实际上给我们提供了很多 函数式接口(例如Runnable、Callable、Comparator),只不过我们很少提及这个概念,Java8及后面高版本中有很多新的函数式接口,例如 Consumer、Function 、Predicate

Predicate:提供了一个test的抽象方法,如果你的lambda表达式,是想返回一个boolean,那可以用这个方法接收

image.png

代码实例如下:

Predicate<OrderInfo> non = orderInfo -> orderInfo.getOrderAmt().compareTo(BigDecimal.valueOf(100)) > 0;
for (OrderInfo orderInfo : orderList)
        if (non.test(orderInfo)) { }

Consumer:该接口定义个一个 accept的抽象方法,没有返回值(void),可以接收任意对象,并对这个对象执行操作

image.png

run(orderInfoList, orderInfo -> { System.out.println("落库"); });
    static void run(List<OrderInfo> orderInfoList, Consumer<OrderInfo> consumer) {
        for (OrderInfo orderInfo : orderInfoList)
            consumer.accept(orderInfo);
    }

Function:接口定义了 apply的,接收任意对象,返回任意对象,使用的时候可以将输入对象的信息映射到输出

image.png

List<CouponInfo> build = build(orderInfoList, orderInfo -> new CouponInfo(orderInfo));
    System.out.println(build.get(0).toString());
    /**
     * 通过订单信息,构件订单使用的优惠信息
     */
    public static List<CouponInfo> build(List<OrderInfo> orderInfoList, Function<OrderInfo, CouponInfo> function) {
        System.out.println("构件CouponInfo");
        List<CouponInfo> couponInfos = new ArrayList<>(orderInfoList.size());
        for (OrderInfo orderInfo : orderInfoList) {
            couponInfos.add(function.apply(orderInfo));
        }
        return couponInfos;
    }
    // 输出:构件CouponInfo  > CouponInfo{orderId='***', orderAmt=***}

小结

本文讲解lambda表达式语法、语义,讲解函数式接口是什么,作用是什么,Lambda表达式的编译原理,以及Java8及高版本的函数式接口Consumer、Function 、Predicate,将会在工作中大量用到

相关文章
|
5天前
|
Java 程序员 C#
Lambda表达式:简洁而强大的函数式编程利器
【4月更文挑战第3天】本文探讨了Lambda表达式的基础和在编程中的应用,包括简化代码和提高可读性。Lambda表达式是匿名函数,用于简单的功能,如示例中的平方运算和列表筛选。通过`map`、`filter`等函数,Lambda表达式能有效处理列表操作。文中还展示了Lambda表达式的高级用法,如闭包特性、异常处理及与高阶函数的结合。通过实例,读者可以学习如何利用Lambda表达式实现更高效、简洁的编程。
27 0
|
2月前
|
设计模式 Java 程序员
为什么程序员喜欢用Lambda表达式?
“lambda 表达式”是一段可以传递的代码,因此它可以被执行一次或多次。在学习语法(甚至包括一些奇怪的术语)之前,我们先回顾一下之前在Java 中一直使用的相似的代码块。
21 0
|
4月前
|
算法 编译器 C++
C++11 Lambda表达式的用法与原理
C++11 Lambda表达式的用法与原理
57 0
|
5月前
|
Java API Scala
【JDK新特性】一篇搞懂Lambda表达式 & 函数式接口
【JDK新特性】一篇搞懂Lambda表达式 & 函数式接口
|
8月前
|
Oracle JavaScript Java
函数式编程与Lambda表达式
函数式编程与Lambda表达式
|
8月前
|
安全 Java
5.3 Lambda表达式在框架和库中的应用:在并发编程中使用Lambda表达式
5.3 Lambda表达式在框架和库中的应用:在并发编程中使用Lambda表达式
38 0
|
8月前
|
Java 编译器
1.2 Lambda表达式的基础:Lambda表达式的函数式接口
1.2 Lambda表达式的基础:Lambda表达式的函数式接口
39 0
|
8月前
|
并行计算 Java 开发者
3.1 函数式接口与Lambda表达式的实际应应用:使用Lambda表达式实现多线程编程
3.1 函数式接口与Lambda表达式的实际应应用:使用Lambda表达式实现多线程编程
54 0
|
8月前
|
自然语言处理 Java 开发者
5.2 Lambda表达式在框架和库中的应用:Lambda表达式在Web开发中的应用
5.2 Lambda表达式在框架和库中的应用:Lambda表达式在Web开发中的应用
55 0
|
9月前
|
Java
lambda表达式的进阶过程(二)
lambda表达式的进阶过程
44 0
lambda表达式的进阶过程(二)

热门文章

最新文章