首页 > 其他分享 >面试官:说说Lambda表达式底层原理?

面试官:说说Lambda表达式底层原理?

时间:2024-09-04 17:27:17浏览次数:11  
标签:面试官 函数 接口 t1 匿名 表达式 Lambda

Lambda 表达式是 Java 8 引入的一种简洁的表示匿名方法的方式,使用它可以用于替代某些匿名内部类对象,从而让程序更简洁,可读性更好。但 Lambda 表达式的底层是如何实现的呢?接下来我们一起来看。

1.未Lambda表达式

未使用 Lambda 表达式之前,我们创建一个线程,可以这样写:

Thread t1 = new Thread(new Runnable() {
    @Override
    public void run() {
        System.out.println("t1");
    }
});
t1.start();

其中 Runnable 匿名内部类,查看 Runnable 源码,我们可以看到 Runnable 的实现如下:

1.1 什么是匿名内部类?

匿名内部类是在 Java 中定义的一个没有名称的内部类。它通常在一个类的成员位置或者方法体内直接定义,并且立即实例化。

匿名内部类的主要用途在于简化代码,避免为了实现一个简单的功能而定义一个完整的类。它特别适用于只需要一次使用的类,比如实现一个接口的单方法(即函数式接口)的场合。

PS:自从 Java 8 引入 Lambda 表达式后,很多原本使用匿名内部类的地方可以被更简洁的 Lambda 表达式替代。

上面代码中的 new Runnable 就是一个标准匿名内部类的使用。

1.2 什么是@FunctionalInterface?

@FunctionalInterface 是 Java 8 引入的一个注解,它用于标记一个接口为函数式接口

函数式接口是指只包含一个抽象方法的接口。这个注解虽然不是必需的,但它提供了一种明确的方式告诉编译器和开发者,这个接口是设计为函数式接口的。

@FunctionalInterface 注解的作用如下:

  1. 编译时检查:当一个接口被标记为 @FunctionalInterface 时,编译器会检查该接口是否只有一个抽象方法。如果不符合函数式接口的定义(即存在多个抽象方法),编译器会报错,提醒开发者修正。这为开发者提供了明确的编译时保障,确保所标记的接口确实符合函数式接口的要求。
  2. 代码明确性:即使不加 @FunctionalInterface 注解,只要接口符合函数式接口的定义,它仍然可以被视为函数式接口。但注解的存在增加了代码的明确性和可读性,使得其他开发者更容易理解该接口的设计意图。
  3. 支持 Lambda 表达式:函数式接口的主要目的是为了支持 Lambda 表达式。通过 Lambda 表达式,开发者可以以更简洁的方式实现函数式接口的抽象方法,从而减少模板代码,使代码更加简洁和易于理解。由于 Lambda 表达式本身不包含类型信息,Java 编译器需要一种机制来确定 Lambda 表达式对应的目标类型。函数式接口就扮演了这一角色——Lambda 表达式可以被赋值给任何兼容的函数式接口类型,编译器会依据接口的唯一抽象方法来推断 Lambda 表达式的参数类型和返回类型。

在 Java 标准库中,有许多使用 @FunctionalInterface 注解的接口,如 java.util.function 包下的 Function、Predicate、Consumer 等,这些接口都是函数式接口,广泛用于数据处理、过滤、转换等操作。此外,在 Spring Boot 框架中,也经常使用函数式接口来定义事件监听器、回调函数等。

2.使用Lambda表达式

未使用 Lambda 表达式之前,我们创建一个线程是这样写的:

Thread t1 = new Thread(new Runnable() {
    @Override
    public void run() {
        System.out.println("t1");
    }
});
t1.start();

而用了 Lambda 表达式,我们可以这样写:

Thread t1 = new Thread(() -> { System.out.println("t1"); });
t1.start();

从上述代码可以看出,当我们使用 Lambda 表达式之后,代码就变得更简洁和优雅了。

3.Lambda详解

Lambda 表达式的语法形式如下:

(parameters) -> expression

或者是:

(parameters) -> { statements; }

以上语法含义如下:

  1. 参数列表:在圆括号内的部分,用于定义传递给 Lambda 体的参数。参数列表可以为空,也可以包含多个参数,参数之间用逗号隔开。
  2. 箭头符号是 Lambda 表达式的分隔符,将参数列表与表达式或语句块分隔开。
  3. Lambda 体:包含了具体的执行逻辑,可以是一个表达式或是一个由多个语句组成的代码块。

3.1 使用场景

Lambda 表达式主要用于执行函数式接口(Function Interface),即只有一个抽象方法的接口。常见的函数式接口包括 java.util.function 包下的 Predicate、Function、Consumer 等。

3.2 举个例子

假设我们有一个List,并且我们想要对这个列表进行过滤操作,只保留偶数元素。使用 Lambda 表达式可以非常方便地实现这一功能:

import java.util.Arrays;
import java.util.List;
import java.util.stream.Collectors;

public class Main {
    public static void main(String[] args) {
        List<Integer> numbers = Arrays.asList(1, 2, 3, 4, 5, 6, 7, 8, 9);
        // 使用 Lambda 表达式过滤出偶数
        List<Integer> evenNumbers = numbers.stream()
                                           .filter(n -> n % 2 == 0)
                                           .collect(Collectors.toList());
        System.out.println(evenNumbers); // 输出 [2, 4, 6, 8]
    }
}

在这个例子中,n -> n % 2 == 0 是一个 Lambda 表达式,它接受一个整数 n 作为输入参数,并返回一个布尔值。这个 Lambda 表达式被用作 filter 方法的参数,该方法期望一个 Predicate 类型的函数式接口实例。

4.Lambda底层原理

Lambda 底层运行原理如下:

  1. 在程序运行时,会在类中生成一个匿名内部类,匿名内部类会实现接口,并重写接口中的抽象方法。
  2. 类中会生成一个静态方法,静态方法中的代码就是 Lambda 表达式中的代码。
  3. 匿名内部类重写的抽象方法,会调用上一步的静态方法,从而实现 Lambda 代码的执行。

所以,综合来说,Lambda 表达式其实是匿名内部类的语法糖,这个语法糖在程序执行时会进行兑现,也就是生成匿名内部类并进行任务执行。

课后思考

使用 javap 能否看出 Lambda 表达式的执行原理呢?说说具体的操作步骤。

本文已收录到我的面试小站 www.javacn.site,其中包含的内容有:Redis、JVM、并发、并发、MySQL、Spring、Spring MVC、Spring Boot、Spring Cloud、MyBatis、设计模式、消息队列等模块。

标签:面试官,函数,接口,t1,匿名,表达式,Lambda
From: https://www.cnblogs.com/vipstone/p/18396973

相关文章

  • 20240907_051745 python 正则表达式 常见元字符
    •.:匹配任意单个字符•\d:匹配数字(等价于[0-9])•\w:匹配字母、数字、下划线(等价于[a-zA-Z0-9_])•\s:匹配空格、制表符、换行符等空白字符•^:匹配开头•$:匹配结尾•*:匹配前面的字符零次或多次•+:匹配前面的字符一次或多次•?:匹配前面的字符零次或一次•[]:匹配方括......
  • 20240907_061745 python 正则表达式 re.match方法
    情况一从头匹配匹配成功的数据可以通过匹配的对象的group()方法获取关注一下匹配不成功的情况情况二从中间匹配......
  • 正则表达式
    什么是正则表达式:正则表达式就是为处理大量的字符串及文本而定义的一套规则和方法。正则表达式就是把人类想要查询的东西,用计算机能识别的语言表达出来的一种规则。正则表达式仅受三剑客(grep,sed,awk)命令支持,其他命无法使用        ^ //以什么开头;"^......
  • 安全: nftables:用describe得到表达式的信息
    例子:ctstate[root@fedora~]#nftdescribectstatectexpression,datatypect_state(conntrackstate)(basetypebitmask,integer),32bitspre-definedsymbolicconstants(inhexadecimal):invalid0x00000001new......
  • C++入门基础知识48——【关于C++函数】之Lambda 函数与表达式
    成长路上不孤单......
  • 11.吐血整理sed入门到精通,sed语法,脚本命令,打印,替换,删除,插入,行替换,字符替换,保
    文章目录前言sed介绍1.sed介绍2.sed语法介绍3.sed脚本命令1.打印2.s替换3.删除脚本命令d3.插入脚本命令a/i4.行替换脚本命令c4.字符替换脚本y5.保存内容脚本w6.插入其他文本r6.中断退出脚本命令q脚本命令当中的地址[address]正则表达式sed[选项]1.sed-i选项2.sed-e......
  • 常见算法和lambda的使用
    常见的七种查找算法:数据结构是数据存储的方式,算法是数据计算的方式。所以在开发中,算法和数据结构息息相关。今天的讲义中会涉及部分数据结构的专业名词,如果各位铁粉有疑惑,可以先看一下哥们后面录制的数据结构,再回头看算法。1.基本查找也叫做顺序查找说明:顺序查找适合于存储结......
  • 正则表达式_java
    今日内容正则表达式教学目标能够理解正则表达式的作用能够使用正则表达式的字符类能够使用正则表达式的逻辑运算符能够使用正则表达式的预定义字符类能够使用正则表达式的限定符能够使用正则表达式的分组能够在String的split方法中使用正则表达式正则表达式......
  • Python正则表达式替换(sub)中如何使用替换函数
    defsub_the_chinese_colon_of_the_question_number(_lines:str)->str:""":param_lines:清理后的文本文件内容功能:查找并替换题号后的中文冒号为英文冒号。"""#定义替换函数,用于调用re.sub:defreplacement(match)->str:"&q......
  • 定义一个正则表达式,使用finditer从表达式内取值后存储到列表中
    演示代码:responce=requests.get(url=url_web,headers=head)url_obj=re.compile(r'<aclass="media-content"target="_blank"href="(?P<url>.*?)"title=".*?"',re.S)list_url=url_obj.finditer(respo......