Java || 你还在遍历搜索集合?别逗了,Java 8 一行代码解决,是真的优雅

简介: Java || 你还在遍历搜索集合?别逗了,Java 8 一行代码解决,是真的优雅

背景

 

是的,如果你想搜索 List 集合,在 Java 8 之前可以使用自身的 contains/ indexOf 方法来查找元素,但仅限是完整元素,而不能模糊搜索或者自定义搜索,这时候确实只能遍历。

但现在都 2021 年了,你还在使用传统的遍历集合的方式搜索 List 集合元素吗?

那你就太 out 了,使用 Java 8 中的 Stream 搜索元素,一行代码即可搞定,而且还真的很优雅!这篇不会介绍 Stream 基础,Stream 系列我之前写过一个专题了,不懂的关注公众号Java技术栈,然后在公众号 Java 教程菜单中阅读。

Stream 搜索

在 Java 8 中,可以将  List 集合转换成 Stream,Stream 提供了一系列强大的搜索功能,比如:filter、find*、*Match 等方法,一行代码就能搞定搜索。

比如现在有初始数据:

public static List<User> list = new ArrayList<>();
/**
 * @author: 栈长
 * @from: 公众号Java技术栈
 */
@BeforeEach
public void initList() {
    list.add(new User("公众号Java技术栈-Petty", 22, 1));
    list.add(new User("公众号Java技术栈-Tom", 38, 1));
    list.add(new User("公众号Java技术栈-Jessica", 43, 0));
    list.add(new User("公众号Java技术栈-John", 15, 1));
    list.add(new User("公众号Java技术栈-Lily", 25, 0));
    list.add(new User("公众号Java技术栈-Lambs", 28, 0));
    list.add(new User("公众号Java技术栈-Jack", 45, 1));
    list.add(new User("公众号Java技术栈-Addy", 9, 0));
    list.add(new User("公众号Java技术栈-Bob", 61, 1));
    list.add(new User("公众号Java技术栈-Candy", 26, 0));
}

用户信息分别为:姓名、年龄、性别。

filter(过滤)

使用 filter 方法实现自定义搜索,比如搜索 List 集合中 所有姓名含有 c 的人:

/**
 * 集合过滤
 * @author: 栈长
 * @from: 公众号Java技术栈
 */
@Test
public void filter() {
    System.out.println("搜索所有姓名含有 c 的人");
    list.stream().filter(u -> u.getName().contains("c")).forEach(System.out::println);
}

输出结果:

findFirst(查找第一个)

查找 Stream 中的第一个元素,比如搜索 List 集合中 第一个年经大于 30 的人:

/**
 * 集合搜索第一个
 * @author: 栈长
 * @from: 公众号Java技术栈
 */
@Test
public void findFirst() {
    System.out.println("搜索第一个年经大于 30 的人");
    User user = list.stream().filter(u -> u.getAge() > 30).findFirst().get();
    System.out.println(user);
}

输出结果:

3ca9e3eecdde2c68f4630098aa803ab.png

示例是需要先 filter 再 findFirst ,但如果不要条件的话,filter 就不是必须的。

findAny(查找任意一个)

查找 Stream 中的任意一个元素,比如搜索 List 集合中 任意一个年经大于 30 的人:

/**
 * 集合搜索任意一个
 * @author: 栈长
 * @from: 公众号Java技术栈
 */
@Test
public void findAny() {
    System.out.println("搜索任意一个年经大于 30 的人");
    User user = list.stream().filter(u -> u.getAge() > 30).findAny().get();
    System.out.println(user.getName());
}

输出结果:

75896ddf00c4f23e04cedef3e016abf.png

为什么结果和 findFirst 一样?和 findFirst 有什么区别呢?

findAny 是查找任意一个元素,在串行流中如果数据较少,一般会返回第一个元素,但在并行流中它返回的结果就不确定了,它可能是流中的任何元素。

findAny 的目的是为了提高并行流操作的性能,但如果需要一个固定的结果,建议使用 findFirst。

anyMatch(任意匹配)

查找 Stream 中的元素是否存在任意匹配,比如搜索 List 集合中 是否存在 XX 人:

/**
 * 集合匹配任意元素
 * @author: 栈长
 * @from: 公众号Java技术栈
 */
@Test
public void anyMatch() {
    System.out.println("是否存在 Jack:" + list.stream().anyMatch(u -> u.getName().contains("Jack")));
    System.out.println("是否存在 Jet:" + list.stream().anyMatch(u -> u.getName().contains("Jet")));
}

输出结果:9101f48abc78aeb404d23184f82077c.png


*Match 返回的结果是 boolean 类型。

noneMatch(空匹配)

查找 Stream 中的元素是否不存在匹配,比如搜索 List 集合中 是否不存在 XX 人:

/**
 * 集合不匹配任意元素
 * @author: 栈长
 * @from: 公众号Java技术栈
 */
@Test
public void noneMatch() {
    System.out.println("是否不存在 Jack:" + list.stream().noneMatch(u -> u.getName().contains("Jack")));
    System.out.println("是否不存在 Jet:" + list.stream().noneMatch(u -> u.getName().contains("Jack")));
}

输出结果:2c6de134b2d1d6f425604d2ba2defb6.png


这个方法的作用就和 anyMatch 相反。

allMatch(全匹配)

查找 Stream 中的元素是否全部匹配,比如搜索 List 集合中 所有人的年纪是否都大于 XX:

/**
 * 集合匹配全部元素
 * @author: 栈长
 * @from: 公众号Java技术栈
 */
@Test
public void allMatch() {
    System.out.println("所有人的年纪都大于3:" + list.stream().allMatch(u -> u.getAge() > 2));
    System.out.println("所有人的年纪都大于30:" + list.stream().allMatch(u -> u.getAge() > 30));
}

输出结果:ea3b07c284d32e5a3c4676d01983948.png


总结

以上所有搜索操作一行代码就能搞定,是不是很简单优雅?

对于 List 之外的集合都可以转换为 List,再转换为 Stream 再进行搜索操作,对于 Stream,搜索简直就是小儿科,你学废用了吗?

赶紧发给身边的同事看看吧,让你们的代码更优雅!

本文所有完整示例源代码已经上传:

https://github.com/javastacks/javastack

欢迎 Star 学习,后面 Java 示例都会在这上面提供!

目录
相关文章
|
5天前
|
Java
java小工具util系列4:基础工具代码(Msg、PageResult、Response、常量、枚举)
java小工具util系列4:基础工具代码(Msg、PageResult、Response、常量、枚举)
16 5
|
3天前
|
Java 开发者
探索Java中的Lambda表达式:简化你的代码之旅##
【8月更文挑战第62天】 Java 8的发布为开发者带来了诸多新特性,其中最引人注目的无疑是Lambda表达式。这一特性不仅让代码变得更加简洁,还极大地提升了开发的效率。本文将通过实际示例,展示如何利用Lambda表达式来优化我们的代码结构,同时探讨其背后的工作原理和性能考量。 ##
|
6天前
|
Java API 开发者
探索Java中的Lambda表达式:简化代码,提升效率
【9月更文挑战第27天】在Java 8中引入的Lambda表达式为编程带来了革命性的变化。通过简洁的语法和强大的功能,它不仅简化了代码编写过程,还显著提升了程序的执行效率。本文将深入探讨Lambda表达式的本质、用法和优势,并结合实例演示其在实际开发中的应用。无论你是Java新手还是资深开发者,都能从中获得启发,优化你的代码设计。
|
6天前
|
域名解析 分布式计算 网络协议
java遍历hdfs路径信息,报错EOFException
java遍历hdfs路径信息,报错EOFException
18 3
|
6天前
|
存储 Java 索引
使用java代码实现左右括号查找
使用java代码实现左右括号查找
|
2月前
|
Java 数据安全/隐私保护
Java代码的执行顺序和构造方法
构造方法是类的一种特殊方法,用于初始化新对象。在 Java 中,每个类默认都有一个与类名同名的构造方法,无需返回类型。构造方法不能用 static、final、synchronized、abstract 或 native 修饰。它可以重载,通过不同的参数列表实现多种初始化方式。构造方法在对象实例化时自动调用,若未显式声明,默认提供一个无参构造方法。构造代码块和静态代码块分别用于对象和类的初始化,按特定顺序执行。
22 0
|
4月前
|
Java
Java代码的执行顺序
Java代码的执行顺序
23 1
|
Java
Java基础-代码执行顺序(重要)
Java代码初始化顺序:     1.由 static 关键字修饰的(如:类变量(静态变量)、静态代码块)将在类被初始化创建实例对象之前被初始化,而且是按顺序从上到下依次被执行。静态(类变量、静态代码块)属于类本身,不依赖于类的实例。     2.没有 static 关键字修饰的(如:实例变量(非静态变量)、非静态代码块)初始化实际上是会被提取到类的构造器中被执行的,但是会比类构造器中的代码
2347 1
LearnJava(四) | Java代码块执行顺序测试
最近笔试常常遇到考察Java代码块执行顺序的题目,网上查看博客错漏百出,特地自己测试了一下。 如有错漏,希望路过的大佬指出来,以便我进行更改。   先上代码吧! public class ClassA { private static St...
944 0
|
Java 机器学习/深度学习
下一篇
无影云桌面