【Java】Stream流、方法引用(二)

简介: 本期主要介绍Stream流、方法引用

1.5 练习:集合元素处理(传统方式)


题目

现在有两个 ArrayList 集合存储队伍当中的多个成员姓名,要求使用传统的 for 循环(或增强 for 循环) 依次 进行以

下若干操作步骤:

1. 第一个队伍只要名字为 3 个字的成员姓名;存储到一个新集合中。

2. 第一个队伍筛选之后只要前 3 个人;存储到一个新集合中。

3. 第二个队伍只要姓张的成员姓名;存储到一个新集合中。

4. 第二个队伍筛选之后不要前 2 个人;存储到一个新集合中。

5. 将两个队伍合并为一个队伍;存储到一个新集合中。

6. 根据姓名创建 Person 对象;存储到一个新集合中。

7. 打印整个队伍的 Person 对象信息。

两个队伍(集合)的代码如下:

image.png

image.png

image.png

解答

既然使用传统的for循环写法,那么:

image.png

运行结果为:

image.png

1.6 练习:集合元素处理(Stream方式)

题目

将上一题当中的传统 for 循环写法更换为 Stream 流式处理方式。两个集合的初始内容不变, Person类的定义也不变

解答

image.png

image.png

第二章 方法引用


在使用Lambda表达式的时候,我们实际上传递进去的代码就是一种解决方案:拿什么参数做什么操作。那么考虑一种情况:如果我们在Lambda中所指定的操作方案,已经有地方存在相同方案,那是否还有必要再写重复逻辑?

2.1 冗余的Lambda场景


来看一个简单的函数式接口以应用Lambda表达式:

image.png

Printable 接口当中唯一的抽象方法 print 接收一个字符串参数,目的就是为了打印显示它。那么通过Lambda来使用它的代码很简单:

image.png

其中 printString 方法只管调用 Printable 接口的 print 方法,而并不管 print 方法的具体实现逻辑会将字符串打印到什么地方去。而 main 方法通过 Lambda 表达式指定了函数式接口 Printable 的具体操作方案为: 拿到

String(类型可推导,所以可省略)数据后,在控制台中输出它

2.2 问题分析


这段代码的问题在于,对字符串进行控制台打印输出的操作方案,明明已经有了现成的实现,那就是 System.out对象中的 println(String) 方法。既然 Lambda 希望做的事情就是调用 println(String) 方法,那何必自己手动调用呢?

2.3 用方法引用改进代码


能否省去Lambda的语法格式(尽管它已经相当简洁)呢?只要引用过去就好了:

image.png

2.4 方法引用符


双冒号 :: 为引用运算符,而它所在的表达式被称为方法引用。如果Lambda要表达的函数方案已经存在于某个方法的实现中,那么则可以通过双冒号来引用该方法作为Lambda的替代者。

语义分析

例如上例中, System.out 对象中有一个重载的 println(String) 方法恰好就是我们所需要的。那么对于

printString 方法的函数式接口参数,对比下面两种写法,完全等效:

Lambda 表达式写法: s - > System.out.println(s) ;

方法引用写法: System.out::println

第一种语义是指:拿到参数之后经 Lambda 之手,继而传递给 System.out.println 方法去处理。

第二种等效写法的语义是指:直接让 System.out 中的 println 方法来取代 Lambda 。两种写法的执行效果完全一样,而第二种方法引用的写法复用了已有方案,更加简洁。

注 :Lambda 中 传递的参数 一定是方法引用中 的那个方法可以接收的类型 , 否则会抛出异常

推导与省略

如果使用Lambda,那么根据可推导就是可省略的原则,无需指定参数类型,也无需指定的重载形式——它们都将被自动推导。而如果使用方法引用,也是同样可以根据上下文进行推导。

函数式接口是 Lambda 的基础,而方法引用是 Lambda 的孪生兄弟。

下面这段代码将会调用 println 方法的不同重载形式,将函数式接口改为 int 类型的参数:

image.png

image.png

2.5 通过对象名引用成员方法


这是最常见的一种用法,与上例相同。如果一个类中已经存在了一个成员方法:

image.png

image.png

image.png

2.6 通过类名称引用静态方法


由于在 java.lang.Math 类中已经存在了静态方法 abs ,所以当我们需要通过 Lambda 来调用该方法时,有两种写法。首先是函数式接口:第一种写法是使用Lambda表达式:

image.png

image.png

在这个例子中,下面两种写法是等效的:

Lambda 表达式: n - > Math.abs(n)

方法引用: Math::abs  

2.7 通过super引用成员方法


如果存在继承关系,当Lambda中需要出现super调用时,也可以使用方法引用进行替代。首先是函数式接口:

image.png

最后是子类 Man 的内容,其中使用了Lambda的写法:

image.png

但是如果使用方法引用来调用父类中的 sayHello 方法会更好,例如另一个子类 Woman

image.png

在这个例子中,下面两种写法是等效的:

Lambda表达式: () -> super.sayHello()

方法引用: super::sayHello

2.8 通过this引用成员方法


this 代表当前对象,如果需要引用的方法就是当前类中的成员方法,那么可以使用 “ this:: 成员方法 ”的格式来使用方法引用。首先是简单的函数式接口:

下面是一个丈夫 Husband 类:

image.png

image.png

开心方法 beHappy 调用了结婚方法 marry ,后者的参数为函数式接口 Richable ,所以需要一个Lambda 表达式。

但是如果这个 Lambda 表达式的内容已经在本类当中存在了,则可以对 Husband 丈夫类进行修改:

image.png

如果希望取消掉Lambda表达式,用方法引用进行替换,则更好的写法为:

image.png

在这个例子中,下面两种写法是等效的:

Lambda 表达式: () - > this.buyHouse()

方法引用: this::buyHouse

2.9 类的构造器引用


由于构造器的名称与类名完全一样,并不固定。所以构造器引用使用 类名称 ::new 的格式表示。首先是一个简单的 Person 类:

image.png

image.png

image.png

在这个例子中,下面两种写法是等效的:

Lambda 表达式: name - > new Person(name)

方法引用: Person::new  

2.10 数组的构造器引用


数组也是 Object 的子类对象,所以同样具有构造器,只是语法稍有不同。如果对应到 Lambda 的使用场景中时,

需要一个函数式接口:

image.png

image.png

在这个例子中,下面两种写法是等效的:

Lambda表达式: length -> new int[length]

方法引用: int[]::new

相关文章
|
2月前
|
消息中间件 Java Kafka
在Java中实现分布式事务的常用框架和方法
总之,选择合适的分布式事务框架和方法需要综合考虑业务需求、性能、复杂度等因素。不同的框架和方法都有其特点和适用场景,需要根据具体情况进行评估和选择。同时,随着技术的不断发展,分布式事务的解决方案也在不断更新和完善,以更好地满足业务的需求。你还可以进一步深入研究和了解这些框架和方法,以便在实际应用中更好地实现分布式事务管理。
|
2月前
|
Java
java小工具util系列5:java文件相关操作工具,包括读取服务器路径下文件,删除文件及子文件,删除文件夹等方法
java小工具util系列5:java文件相关操作工具,包括读取服务器路径下文件,删除文件及子文件,删除文件夹等方法
84 9
|
1月前
|
存储 Java 数据挖掘
Java 8 新特性之 Stream API:函数式编程风格的数据处理范式
Java 8 引入的 Stream API 提供了一种新的数据处理方式,支持函数式编程风格,能够高效、简洁地处理集合数据,实现过滤、映射、聚合等操作。
62 6
|
1月前
|
Java API 开发者
Java中的Lambda表达式与Stream API的协同作用
在本文中,我们将探讨Java 8引入的Lambda表达式和Stream API如何改变我们处理集合和数组的方式。Lambda表达式提供了一种简洁的方法来表达代码块,而Stream API则允许我们对数据流进行高级操作,如过滤、映射和归约。通过结合使用这两种技术,我们可以以声明式的方式编写更简洁、更易于理解和维护的代码。本文将介绍Lambda表达式和Stream API的基本概念,并通过示例展示它们在实际项目中的应用。
|
2月前
|
安全 Java 开发者
Java中WAIT和NOTIFY方法必须在同步块中调用的原因
在Java多线程编程中,`wait()`和`notify()`方法是实现线程间协作的关键。这两个方法必须在同步块或同步方法中调用,这一要求背后有着深刻的原因。本文将深入探讨为什么`wait()`和`notify()`方法必须在同步块中调用,以及这一机制如何确保线程安全和避免死锁。
47 4
|
2月前
|
Java
深入探讨Java中的中断机制:INTERRUPTED和ISINTERRUPTED方法详解
在Java多线程编程中,中断机制是协调线程行为的重要手段。了解和正确使用中断机制对于编写高效、可靠的并发程序至关重要。本文将深入探讨Java中的`Thread.interrupted()`和`Thread.isInterrupted()`方法的区别及其应用场景。
68 4
|
2月前
|
Java 数据处理 数据安全/隐私保护
Java处理数据接口方法
Java处理数据接口方法
28 1
|
29天前
|
Rust 安全 Java
Java Stream 使用指南
本文介绍了Java中Stream流的使用方法,包括如何创建Stream流、中间操作(如map、filter、sorted等)和终结操作(如collect、forEach等)。此外,还讲解了并行流的概念及其可能带来的线程安全问题,并给出了示例代码。
|
2月前
|
存储 Java 程序员
Java基础的灵魂——Object类方法详解(社招面试不踩坑)
本文介绍了Java中`Object`类的几个重要方法,包括`toString`、`equals`、`hashCode`、`finalize`、`clone`、`getClass`、`notify`和`wait`。这些方法是面试中的常考点,掌握它们有助于理解Java对象的行为和实现多线程编程。作者通过具体示例和应用场景,详细解析了每个方法的作用和重写技巧,帮助读者更好地应对面试和技术开发。
140 4
|
2月前
|
安全 Java API
Java中的Lambda表达式与Stream API的高效结合####
探索Java编程中Lambda表达式与Stream API如何携手并进,提升数据处理效率,实现代码简洁性与功能性的双重飞跃。 ####
32 0