当前位置: 首页 > news >正文

深入学习Java中的Lambda表达式

深入学习Java中的Lambda表达式

自Java 8引入以来,Lambda表达式彻底改变了Java的编程风格,让代码变得更加简洁、易读,尤其是在函数式编程的场景中。接下来,我们将深入探讨Lambda表达式的语法、原理以及实际应用,帮助你更好地理解并运用这一强大的工具。

文章目录

  • 深入学习Java中的Lambda表达式
    • 1. Lambda表达式的基本语法
      • 1.1 无参数的Lambda表达式
      • 1.2 一个参数的Lambda表达式
      • 1.3 多个参数的Lambda表达式
    • 2. Lambda表达式的原理
      • 2.1 Lambda表达式的实现机制:函数式接口
      • 2.2 编译器生成的匿名类
      • 2.3 方法引用(Method References)
      • 2.4 Lambda表达式与接口方法的绑定
      • 2.5 JVM与Lambda的实现:Invokedynamic
      • 2.6 Lambda表达式与性能
    • 3. Lambda表达式的应用
      • 3.1 配合Stream API处理集合数据
      • 3.2 Lambda表达式简化匿名内部类
    • 4. 总结

1. Lambda表达式的基本语法

(parameters) -> expression

或者如果需要更复杂的代码块:

(parameters) -> { statements }
  • parameters:Lambda表达式的输入参数,可以有一个或多个。若只有一个参数,可以省略圆括号;如果没有参数或有多个参数,必须使用圆括号。
  • ->:箭头符号,用于分隔参数和表达式。
  • expression:Lambda体,即表达式或代码块,定义了Lambda的实现逻辑。如果只有一个表达式,Java会自动返回该表达式的值;如果是一个代码块,则必须显式地使用return语句。

1.1 无参数的Lambda表达式

如果没有参数,必须使用圆括号。

Runnable r = () -> System.out.println("Hello, Lambda!");
r.run(); // 输出 "Hello, Lambda!"
  • Runnable是一个没有输入参数的函数接口,因此我们使用()表示没有参数,->后跟要执行的代码块。

1.2 一个参数的Lambda表达式

如果只有一个参数,可以省略括号。

List<String> names = Arrays.asList("Alice", "Bob", "Charlie");names.forEach(name -> System.out.println(name));  // 输出每个名字
  • 这里的Lambda表达式接收一个name参数,并执行System.out.println(name)。它是forEach方法的一个参数,用来对列表中的每个元素进行操作。

1.3 多个参数的Lambda表达式

当有多个参数时,必须使用括号。

List<Integer> numbers = Arrays.asList(1, 2, 3, 4, 5);int sum = numbers.stream().reduce(0, (a, b) -> a + b);  // 求和操作
System.out.println(sum);  // 输出 15
  • reduce方法接收两个参数,第一个是初始值0,第二个是Lambda表达式(a, b) -> a + b,它将集合中的所有数字相加。

2. Lambda表达式的原理

2.1 Lambda表达式的实现机制:函数式接口

Lambda表达式主要应用于函数式接口,即只包含一个抽象方法的接口。例如:

@FunctionalInterface
public interface MyFunction {int apply(int a, int b);
}

Lambda表达式在Java中通常是与函数式接口(只有一个抽象方法的接口)结合使用的。Java在编译时会将Lambda表达式转换为该接口的实现,并使用动态代理机制生成实现类。

2.2 编译器生成的匿名类

Lambda表达式并不是直接创建一个类,而是通过一种称为匿名类的机制在幕后创建。事实上,Lambda表达式的内部实现往往会通过编译器生成一个匿名类或方法的形式来实现接口的抽象方法。

例如,以下的Lambda表达式:

(a, b) -> a + b

会被编译器转换为类似这样的一种形式:

new MyFunction() {@Overridepublic int apply(int a, int b) {return a + b;}
}

但是这种匿名类并不会像传统的匿名类那样显式地出现在代码中,而是由Java的虚拟机(JVM)在运行时动态创建。

2.3 方法引用(Method References)

Lambda表达式可以与方法引用配合使用,进一步简化代码。例如,你可以将一个Lambda表达式转换为对已有方法的引用,从而避免编写冗余代码。

// 使用Lambda表达式
Function<String, Integer> stringLength = s -> s.length();// 使用方法引用
Function<String, Integer> stringLengthMethodRef = String::length;

方法引用本质上是一个对某个方法的引用,JVM会在运行时将其与Lambda表达式的功能关联。

2.4 Lambda表达式与接口方法的绑定

Lambda表达式与接口的抽象方法是通过目标类型推断来绑定的。具体来说,JVM会根据Lambda表达式所使用的接口类型来推断出Lambda表达式的目标类型。

例如,考虑以下代码:

List<String> names = Arrays.asList("Tom", "Jerry", "Mickey");
names.forEach(name -> System.out.println(name));

在这段代码中,forEach方法接受一个Consumer<T>消费类型的接口。Lambda表达式name -> System.out.println(name)的目标类型会被推断为Consumer<String>接口的方法accept()

四大核心函数式接口

2.5 JVM与Lambda的实现:Invokedynamic

Java 8引入了**invokedynamic**字节码指令,这是Lambda表达式的核心机制之一。invokedynamic指令允许JVM在运行时动态地绑定和执行代码,这意味着在运行时,Lambda表达式的具体实现才会被确定。

Lambda表达式通过以下几个步骤实现:

  1. 编译阶段:Lambda表达式在编译时会被转换成特定的字节码,其中包括一个方法句柄,该方法句柄指向Lambda表达式所实现的接口方法。
  2. 运行时:JVM通过invokedynamic指令来延迟方法绑定。这使得Lambda表达式的实际调用能够在运行时动态决定,从而提高性能。
  3. 方法引用:Lambda表达式还可以被实现为方法引用。方法引用本质上也是一种对Lambda表达式的优化,因为它不需要通过Lambda表达式生成新的对象或匿名类实例,而是直接引用现有的方法。

2.6 Lambda表达式与性能

Lambda表达式在性能上并不会带来显著的开销。由于Lambda表达式通常会被编译为方法句柄,且通过invokedynamic指令动态绑定方法,JVM的优化使得Lambda表达式的性能与传统的匿名类相比几乎没有差别,甚至可能更高效。

但是需要注意的是,Lambda表达式的实际性能仍然取决于具体的使用场景。例如,在一些频繁调用的场景中,Lambda表达式可能会有一些性能上的开销,但总体来说,这种开销是微乎其微的。

3. Lambda表达式的应用

Lambda表达式在Java中广泛应用,特别是在处理集合、流操作等场景中。通过简洁的语法,Lambda表达式使得代码更加清晰、简洁,同时提升了可读性和可维护性。

3.1 配合Stream API处理集合数据

Stream API中,Lambda表达式非常适合用于处理集合数据的过滤、映射、排序等操作。比如,我们可以使用Lambda表达式配合Stream API筛选出集合中的某些元素。下面是一个示例,展示如何使用Lambda表达式筛选出所有偶数。

List<Integer> numbers = Arrays.asList(1, 2, 3, 4, 5, 6, 7, 8, 9, 10);// 使用Lambda表达式过滤出偶数
numbers.stream().filter(n -> n % 2 == 0)  // 过滤出偶数.forEach(System.out::println);  // 输出结果
  • numbers.stream():将numbers列表转化为一个Stream流,Stream流支持更丰富的函数式操作。
  • .filter(n -> n % 2 == 0):这是一个Lambda表达式,用于过滤出所有偶数。filter方法会遍历流中的每个元素,只有符合条件(即n % 2 == 0)的元素会被保留。
  • .forEach(System.out::println):对每一个经过过滤后的元素执行println操作,这里使用方法引用System.out::println,其等价于n -> System.out.println(n),简化了代码。

3.2 Lambda表达式简化匿名内部类

在Java 8之前,匿名内部类是实现接口或抽象类的主要方式。Lambda表达式使得这些代码更加简洁。以下是Comparator排序的匿名内部类和Lambda表达式的对比:

使用匿名内部类:

List<String> names = Arrays.asList("Alice", "Bob", "Charlie");
Collections.sort(names, new Comparator<String>() {@Overridepublic int compare(String s1, String s2) {return s1.compareTo(s2);  // 按字母顺序排序}
});

使用Lambda表达式:

List<String> names = Arrays.asList("Alice", "Bob", "Charlie");
Collections.sort(names, (s1, s2) -> s1.compareTo(s2));  // 按字母顺序排序
  • 匿名内部类代码较长,需要声明一个新的类,并重写compare方法。
  • Lambda表达式简洁,直接用(s1, s2) -> s1.compareTo(s2)来实现Comparatorcompare方法。

4. 总结

Lambda表达式是Java 8引入的一个功能,它通过简洁的语法允许你编写匿名函数,主要用于函数式接口(接口中只有一个抽象方法)的实现,Lambda有着以下特性:

  • 简洁性:Lambda表达式可以让代码更简洁、更直观,减少了冗余的类和方法声明。
  • 函数式编程支持:Lambda表达式支持函数式编程,尤其是和Java 8的Stream API结合使用时,可以进行复杂的数据操作,如过滤、映射、排序等。
  • 延迟执行:在Stream的操作中,Lambda表达式通常是惰性求值的,这意味着只有在最终操作(如forEachcollect等)触发时,数据才会被实际处理。

本文是在GPT的帮助下进行整理与归纳的,如果对你有帮助,欢迎点赞、留言与转发

相关文章:

深入学习Java中的Lambda表达式

深入学习Java中的Lambda表达式 自Java 8引入以来&#xff0c;Lambda表达式彻底改变了Java的编程风格&#xff0c;让代码变得更加简洁、易读&#xff0c;尤其是在函数式编程的场景中。接下来&#xff0c;我们将深入探讨Lambda表达式的语法、原理以及实际应用&#xff0c;帮助你…...

1.2 AI 量化炒股的起源与发展

**定性价值**&#xff1a;AI量化炒股通过算法模型实现投资决策自动化&#xff0c;显著提升交易效率与风险控制能力&#xff0c;打破传统人工交易的主观性与延迟性&#xff0c;推动金融科技向智能化、数据驱动方向迭代&#xff0c;具有颠覆传统投资模式的战略意义。 **定量价值…...

计算机单位之详解——存储单位Byte 网络传输单位bps 视频码率单位bps

前言&#xff1a; 计算机里面单位有点复杂&#xff0c;容易混淆&#xff0c;很多时候混起来就容易概念不理解&#xff0c;包括一些小问题&#xff0c;比如说&#xff1a;为什么我买了1T硬盘&#xff0c;实际存在虚标。为什么所谓的千兆宽带&#xff0c;下载起来没有1G每秒&…...

IDEA关闭SpringBoot程序后仍然占用端口的排查与解决

IDEA关闭SpringBoot程序后仍然占用端口的排查与解决 问题描述 在使用 IntelliJ IDEA 开发 Spring Boot 应用时&#xff0c;有时即使关闭了应用&#xff0c;程序仍然占用端口&#xff08;例如&#xff1a;4001 端口&#xff09;。这会导致重新启动应用时出现端口被占用的错误&a…...

deepseek清华大学第二版 如何获取 DeepSeek如何赋能职场应用 PDF文档 电子档(附下载)

deepseek清华大学第二版 DeepSeek如何赋能职场 pdf文件完整版下载 https://pan.baidu.com/s/1aQcNS8UleMldcoH0Jc6C6A?pwd1234 提取码: 1234 或 https://pan.quark.cn/s/3ee62050a2ac...

【python随手记】——读取文本文件内容转换为json格式

文章目录 前言一、TXT文件转换为JSON数组1.txt文件内容2.python代码3.输出结果 二、TXT文件转换为JSON对象1.txt文件2.python代码3.输出结果 前言 场景&#xff1a;用于读取包含空格分隔数据的TXT文件&#xff0c;并将其转换为结构化JSON文件 一、TXT文件转换为JSON数组 1.tx…...

k8s集群3主5从高可用架构(kubeadm方式安装k8s)

关键步骤说明 环境准备阶段 系统更新&#xff1a;所有节点执行yum/apt update确保软件包最新时间同步&#xff1a;通过ntpdate time.windows.com或部署NTP服务器网络规划&#xff1a;明确划分Service网段&#xff08;默认10.96.0.0/12&#xff09;和Pod网段&#xff08;如Flann…...

基于 sklearn 的均值偏移聚类算法的应用

基于 sklearn 的均值偏移聚类算法的应用 在机器学习和数据挖掘中&#xff0c;聚类算法是一类非常重要的无监督学习方法。它的目的是将数据集中的数据点划分为若干个类&#xff0c;使得同一类的样本点彼此相似&#xff0c;而不同类的样本点相互之间差异较大。均值偏移聚类&…...

三、大模型微调的多种方法与应用场景

详解大模型微调的多种方法与应用场景 随着大模型的不断发展&#xff0c;如何有效地微调这些庞大的预训练模型以适应特定任务成为了研究和应用中的一个重要问题。大模型微调不仅能够提高任务性能&#xff0c;还能在不同的业务需求中提升模型的适应性。在本文中&#xff0c;我们…...

第2课 树莓派镜像的烧录

树莓派的系统通常是安装在SD卡上的‌。SD卡作为启动设备,负责启动树莓派并加载操作系统。这种设计使得树莓派具有便携性和灵活性,用户可以通过更换SD卡来更换操作系统或恢复出厂设置。 烧录树莓派的镜像即是将树莓派镜像烧录到SD卡上,在此期间会格式化SD卡,如果SD卡…...

SQL之order by盲注

目录 一.order by盲注的原理 二.注入方式 a.布尔盲注 b.时间盲注 三.防御 一.order by盲注的原理 order by子句是用于按指定列排序查询结果&#xff0c;列名或列序号皆可。 order by 后面接的字段或者数字不一样&#xff0c;那么这个数据表的排序就会不同。 order by 盲…...

AI大模型(四)基于Deepseek本地部署实现模型定制与调教

AI大模型&#xff08;四&#xff09;基于Deepseek本地部署实现模型定制与调教 DeepSeek开源大模型在榜单上以黑马之姿横扫多项评测&#xff0c;其社区热度指数暴涨、一跃成为近期内影响力最高的话题&#xff0c;这个来自中国团队的模型向世界证明&#xff1a;让每个普通人都能…...

java后端开发day19--学生管理系统升级

&#xff08;以下内容全部来自上述课程&#xff09; 1.要求及思路 1.总体框架 2.注册 3.登录 4.忘记密码 2.代码 1.javabean public class User1 {private String username;private String password;private String personID;private String phoneNumber;public User1() {…...

MFC文件和注册表的操作

MFC文件和注册表的操作 日志、操作配置文件、ini、注册表、音视频的文件存储 Linux下一切皆文件 C/C操作文件 const char* 与 char* const const char* 常量指针&#xff0c;表示指向的内容为常量。指针可以指向其他变量&#xff0c;但是内容不能再变了 char szName[6]&qu…...

vscode如何使用鼠标滚轮调整字体大小

1.打开设置 2.搜索Font Ligatures 3.编辑配置文件 4.修改代码并保存 修改前 修改后 在最后一行添加&#xff1a;“editor.mouseWheelZoom”: true 记得在上一行最后&#xff0c;加上英文版的“,”逗号 5.配置成功&#xff0c;再次按Ctrl鼠标滚轮便可以缩放了。...

C++之vector和list辨析

std::vector 和 std::list 是 C 标准库中两种常用的容器&#xff0c;它们都用于存储和管理元素集合&#xff0c;但在底层实现和性能特性上有显著的区别。 1. 底层实现 std::vector: 基于动态数组实现。元素在内存中是连续存储的。支持随机访问&#xff08;通过下标访问元素&a…...

冯诺依曼体系结构 ──── linux第8课

目录 冯诺依曼体系结构 关于冯诺依曼&#xff0c;必须强调几点&#xff1a; 冯诺依曼体系结构 我们常见的计算机&#xff0c;如笔记本。我们不常见的计算机&#xff0c;如服务器&#xff0c;大部分都遵守冯诺依曼体系 输入单元&#xff1a;包括键盘, 鼠标&#xff0c;网卡,扫…...

EX_25/2/22

找到第一天mystring练习&#xff0c;实现以下功能 mystring str "hello" mystring ptr "world" str str ptr; str ptr str[0] H #include <iostream> #include <cstring> #include <cstdlib> #include <unistd.h> #in…...

rust安装教程以及git连接到远程仓库

1.官方网站下载rustup-init程序 链接: rust-lang 从这里可以获取到rust的下载程序,这个下载程序会帮助你下载visual-studio的安装包从而获取相关的编译环境。 tips:无需再下载visual_studio 2确认安装所需要的框架&#xff0c;SKD工具 安装完毕之后可以检查一下 rustc --ve…...

Kafka系列之:记录一次源头数据库刷数据,造成数据丢失的原因

Kafka系列之:记录一次源头数据库刷数据,造成数据丢失的原因 一、背景二、查看topic日志信息三、结论四、解决方法一、背景 源头数据库在很短的时间内刷了大量的数据,部分数据在hdfs丢失了 理论上debezium数据采集不会丢失,就需要排查数据链路某个节点是否有数据丢失。 数据…...

【SQL学习笔记1】增删改查+多表连接全解析(内附SQL免费在线练习工具)

可以使用Sqliteviz这个网站免费编写sql语句&#xff0c;它能够让用户直接在浏览器内练习SQL的语法&#xff0c;不需要安装任何软件。 链接如下&#xff1a; sqliteviz 注意&#xff1a; 在转写SQL语法时&#xff0c;关键字之间有一个特定的顺序&#xff0c;这个顺序会影响到…...

P3 QT项目----记事本(3.8)

3.8 记事本项目总结 项目源码 1.main.cpp #include "widget.h" #include <QApplication> int main(int argc, char *argv[]) {QApplication a(argc, argv);Widget w;w.show();return a.exec(); } 2.widget.cpp #include "widget.h" #include &q…...

12.找到字符串中所有字母异位词

&#x1f9e0; 题目解析 题目描述&#xff1a; 给定两个字符串 s 和 p&#xff0c;找出 s 中所有 p 的字母异位词的起始索引。 返回的答案以数组形式表示。 字母异位词定义&#xff1a; 若两个字符串包含的字符种类和出现次数完全相同&#xff0c;顺序无所谓&#xff0c;则互为…...

【C++从零实现Json-Rpc框架】第六弹 —— 服务端模块划分

一、项目背景回顾 前五弹完成了Json-Rpc协议解析、请求处理、客户端调用等基础模块搭建。 本弹重点聚焦于服务端的模块划分与架构设计&#xff0c;提升代码结构的可维护性与扩展性。 二、服务端模块设计目标 高内聚低耦合&#xff1a;各模块职责清晰&#xff0c;便于独立开发…...

均衡后的SNRSINR

本文主要摘自参考文献中的前两篇&#xff0c;相关文献中经常会出现MIMO检测后的SINR不过一直没有找到相关数学推到过程&#xff0c;其中文献[1]中给出了相关原理在此仅做记录。 1. 系统模型 复信道模型 n t n_t nt​ 根发送天线&#xff0c; n r n_r nr​ 根接收天线的 MIMO 系…...

算法岗面试经验分享-大模型篇

文章目录 A 基础语言模型A.1 TransformerA.2 Bert B 大语言模型结构B.1 GPTB.2 LLamaB.3 ChatGLMB.4 Qwen C 大语言模型微调C.1 Fine-tuningC.2 Adapter-tuningC.3 Prefix-tuningC.4 P-tuningC.5 LoRA A 基础语言模型 A.1 Transformer &#xff08;1&#xff09;资源 论文&a…...

LINUX 69 FTP 客服管理系统 man 5 /etc/vsftpd/vsftpd.conf

FTP 客服管理系统 实现kefu123登录&#xff0c;不允许匿名访问&#xff0c;kefu只能访问/data/kefu目录&#xff0c;不能查看其他目录 创建账号密码 useradd kefu echo 123|passwd -stdin kefu [rootcode caozx26420]# echo 123|passwd --stdin kefu 更改用户 kefu 的密码…...

Web中间件--tomcat学习

Web中间件–tomcat Java虚拟机详解 什么是JAVA虚拟机 Java虚拟机是一个抽象的计算机&#xff0c;它可以执行Java字节码。Java虚拟机是Java平台的一部分&#xff0c;Java平台由Java语言、Java API和Java虚拟机组成。Java虚拟机的主要作用是将Java字节码转换为机器代码&#x…...

Scrapy-Redis分布式爬虫架构的可扩展性与容错性增强:基于微服务与容器化的解决方案

在大数据时代&#xff0c;海量数据的采集与处理成为企业和研究机构获取信息的关键环节。Scrapy-Redis作为一种经典的分布式爬虫架构&#xff0c;在处理大规模数据抓取任务时展现出强大的能力。然而&#xff0c;随着业务规模的不断扩大和数据抓取需求的日益复杂&#xff0c;传统…...

uniapp 集成腾讯云 IM 富媒体消息(地理位置/文件)

UniApp 集成腾讯云 IM 富媒体消息全攻略&#xff08;地理位置/文件&#xff09; 一、功能实现原理 腾讯云 IM 通过 消息扩展机制 支持富媒体类型&#xff0c;核心实现方式&#xff1a; 标准消息类型&#xff1a;直接使用 SDK 内置类型&#xff08;文件、图片等&#xff09;自…...