广告
返回顶部
首页 > 资讯 > 精选 >Java如何使用 Lambda 表达式实现超强的排序功能
  • 565
分享到

Java如何使用 Lambda 表达式实现超强的排序功能

2023-06-25 14:06:24 565人浏览 泡泡鱼
摘要

这篇文章主要介绍Java如何使用 Lambda 表达式实现超强的排序功能,文中介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们一定要看完!首先,我们定义一个基础类,后面我们将根据这个基础类演示如何在内存中排序。@Data@NoArgsC

这篇文章主要介绍Java如何使用 Lambda 表达式实现超强的排序功能,文中介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们一定要看完!

    首先,我们定义一个基础类,后面我们将根据这个基础类演示如何在内存中排序。

    @Data@NoArgsConstructor@AllArgsConstructorpublic class Student {    private String name;    private int age;    @Override    public boolean equals(Object o) {        if (this == o) {            return true;        }        if (o == null || getClass() != o.getClass()) {            return false;        }        Student student = (Student) o;        return age == student.age && Objects.equals(name, student.name);    }    @Override    public int hashCode() {        return Objects.hash(name, age);    }}

    基于Comparator排序

    在 Java8 之前,我们都是通过实现Comparator接口完成排序,比如:

    new Comparator<Student>() {    @Override    public int compare(Student h2, Student h3) {        return h2.getName().compareTo(h3.getName());    }};

    这里展示的是匿名内部类的定义,如果是通用的对比逻辑,可以直接定义一个实现类。使用起来也比较简单,如下就是应用:

    @Testvoid baseSortedOrigin() {    final List<Student> students = Lists.newArrayList(            new Student("Tom", 10),            new Student("Jerry", 12)    );    Collections.sort(students, new Comparator<Student>() {        @Override        public int compare(Student h2, Student h3) {            return h2.getName().compareTo(h3.getName());        }    });    Assertions.assertEquals(students.get(0), new Student("Jerry", 12));}

    这里使用了 Junit5 实现单元测试,用来验证逻辑非常适合。

    因为定义的Comparator是使用name字段排序,在 Java 中,String类型的排序是通过单字符的 ASCII 码顺序判断的,J排在T的前面,所以Jerry排在第一个。

    使用 Lambda 表达式替换Comparator匿名内部类

    使用过 Java8 的 Lamdba 的应该知道,匿名内部类可以简化为 Lambda 表达式为:

    Collections.sort(students, (Student h2, Student h3) -> h2.getName().compareTo(h3.getName()));

    在 Java8 中,List类中增加了sort方法,所以Collections.sort可以直接替换为:

    students.sort((Student h2, Student h3) -> h2.getName().compareTo(h3.getName()));

    根据 Java8 中 Lambda 的类型推断,我们可以将指定的Student类型简写:

    students.sort((h2, h3) -> h2.getName().compareTo(h3.getName()));

    至此,我们整段排序逻辑可以简化为:

    @Testvoid baseSortedLambdaWithInferring() {    final List<Student> students = Lists.newArrayList(            new Student("Tom", 10),            new Student("Jerry", 12)    );    students.sort((h2, h3) -> h2.getName().compareTo(h3.getName()));    Assertions.assertEquals(students.get(0), new Student("Jerry", 12));}

    通过静态方法抽取公共的 Lambda 表达式

    我们可以在Student中定义一个静态方法:

    public static int compareByNameThenAge(Student s1, Student s2) {    if (s1.name.equals(s2.name)) {        return Integer.compare(s1.age, s2.age);    } else {        return s1.name.compareTo(s2.name);    }}

    这个方法需要返回一个int类型参数,在 Java8 中,我们可以在 Lambda 中使用该方法:

    @Testvoid sortedUsingStaticMethod() {    final List<Student> students = Lists.newArrayList(            new Student("Tom", 10),            new Student("Jerry", 12)    );    students.sort(Student::compareByNameThenAge);    Assertions.assertEquals(students.get(0), new Student("Jerry", 12));}

    借助Comparator的comparing方法

    在 Java8 中,Comparator类新增了comparing方法,可以将传递的Function参数作为比较元素,比如:

    @Testvoid sortedUsinGComparator() {    final List<Student> students = Lists.newArrayList(            new Student("Tom", 10),            new Student("Jerry", 12)    );    students.sort(Comparator.comparing(Student::getName));    Assertions.assertEquals(students.get(0), new Student("Jerry", 12));}

    多条件排序

    我们在静态方法一节中展示了多条件排序,还可以在Comparator匿名内部类中实现多条件逻辑:

    @Testvoid sortedMultiCondition() {    final List<Student> students = Lists.newArrayList(            new Student("Tom", 10),            new Student("Jerry", 12),            new Student("Jerry", 13)    );    students.sort((s1, s2) -> {        if (s1.getName().equals(s2.getName())) {            return Integer.compare(s1.getAge(), s2.getAge());        } else {            return s1.getName().compareTo(s2.getName());        }    });    Assertions.assertEquals(students.get(0), new Student("Jerry", 12));}

    从逻辑来看,多条件排序就是先判断第一级条件,如果相等,再判断第二级条件,依次类推。在 Java8 中可以使用comparing和一系列thenComparing表示多级条件判断,上面的逻辑可以简化为:

    @Testvoid sortedMultiConditionUsingComparator() {    final List<Student> students = Lists.newArrayList(            new Student("Tom", 10),            new Student("Jerry", 12),            new Student("Jerry", 13)    );    students.sort(Comparator.comparing(Student::getName).thenComparing(Student::getAge));    Assertions.assertEquals(students.get(0), new Student("Jerry", 12));}

    这里的thenComparing方法是可以有多个的,用于表示多级条件判断,这也是函数式编程的方便之处。

    在Stream中进行排序

    Java8 中,不但引入了 Lambda 表达式,还引入了一个全新的流式 api:Stream API,其中也有sorted方法用于流式计算时排序元素,可以传入Comparator实现排序逻辑:

    @Testvoid streamSorted() {    final List<Student> students = Lists.newArrayList(            new Student("Tom", 10),            new Student("Jerry", 12)    );    final Comparator<Student> comparator = (h2, h3) -> h2.getName().compareTo(h3.getName());    final List<Student> sortedStudents = students.stream()            .sorted(comparator)            .collect(Collectors.toList());    Assertions.assertEquals(sortedStudents.get(0), new Student("Jerry", 12));}

    同样的,我们可以通过 Lambda 简化书写:

    @Testvoid streamSortedUsingComparator() {    final List<Student> students = Lists.newArrayList(            new Student("Tom", 10),            new Student("Jerry", 12)    );    final Comparator<Student> comparator = Comparator.comparing(Student::getName);    final List<Student> sortedStudents = students.stream()            .sorted(comparator)            .collect(Collectors.toList());    Assertions.assertEquals(sortedStudents.get(0), new Student("Jerry", 12));}

    倒序排列

    调转排序判断

    排序就是根据compareTo方法返回的值判断顺序,如果想要倒序排列,只要将返回值取返即可:

    @Testvoid sortedReverseUsingComparator2() {    final List<Student> students = Lists.newArrayList(            new Student("Tom", 10),            new Student("Jerry", 12)    );    final Comparator<Student> comparator = (h2, h3) -> h3.getName().compareTo(h2.getName());    students.sort(comparator);    Assertions.assertEquals(students.get(0), new Student("Tom", 10));}

    可以看到,正序排列的时候,我们是h2.getName().compareTo(h3.getName()),这里我们直接倒转过来,使用的是h3.getName().compareTo(h2.getName()),也就达到了取反的效果。在 Java 的Collections中定义了一个java.util.Collections.ReverseComparator内部私有类,就是通过这种方式实现元素反转。

    借助Comparatorreversed方法倒序

    在 Java8 中新增了reversed方法实现倒序排列,用起来也是很简单:

    @Testvoid sortedReverseUsingComparator() {    final List<Student> students = Lists.newArrayList(            new Student("Tom", 10),            new Student("Jerry", 12)    );    final Comparator<Student> comparator = (h2, h3) -> h2.getName().compareTo(h3.getName());    students.sort(comparator.reversed());    Assertions.assertEquals(students.get(0), new Student("Tom", 10));}

    在Comparator.comparing中定义排序反转

    comparing方法还有一个重载方法,java.util.Comparator#comparing(java.util.function.Function<? super T,? extends U>, java.util.Comparator<? super U>),第二个参数就可以传入Comparator.reverseOrder(),可以实现倒序:

    @Testvoid sortedUsingComparatorReverse() {    final List<Student> students = Lists.newArrayList(            new Student("Tom", 10),            new Student("Jerry", 12)    );    students.sort(Comparator.comparing(Student::getName, Comparator.reverseOrder()));    Assertions.assertEquals(students.get(0), new Student("Jerry", 12));}

    在Stream中定义排序反转

    Stream中的操作与直接列表排序类似,可以反转Comparator定义,也可以使用Comparator.reverseOrder()反转。实现如下:

    @Testvoid streamReverseSorted() {    final List<Student> students = Lists.newArrayList(            new Student("Tom", 10),            new Student("Jerry", 12)    );    final Comparator<Student> comparator = (h2, h3) -> h3.getName().compareTo(h2.getName());    final List<Student> sortedStudents = students.stream()            .sorted(comparator)            .collect(Collectors.toList());    Assertions.assertEquals(sortedStudents.get(0), new Student("Tom", 10));}@Testvoid streamReverseSortedUsingComparator() {    final List<Student> students = Lists.newArrayList(            new Student("Tom", 10),            new Student("Jerry", 12)    );    final List<Student> sortedStudents = students.stream()            .sorted(Comparator.comparing(Student::getName, Comparator.reverseOrder()))            .collect(Collectors.toList());    Assertions.assertEquals(sortedStudents.get(0), new Student("Tom", 10));}

    null 值的判断

    前面的例子中都是有值元素排序,能够覆盖大部分场景,但有时候我们还是会碰到元素中存在null的情况:

    1. 列表中的元素是 null

    2. 列表中的元素参与排序条件的字段是 null

    如果还是使用前面的那些实现,我们会碰到NullPointException异常,即 NPE,简单演示一下:

    @Testvoid sortedNullGotNPE() {    final List<Student> students = Lists.newArrayList(            null,            new Student("Snoopy", 12),            null    );    Assertions.assertThrows(NullPointerException.class,            () -> students.sort(Comparator.comparing(Student::getName)));}

    所以,我们需要考虑这些场景。

    元素是 null 的笨拙实现

    最先想到的就是判空:

    @Testvoid sortedNullNoNPE() {    final List<Student> students = Lists.newArrayList(            null,            new Student("Snoopy", 12),            null    );    students.sort((s1, s2) -> {        if (s1 == null) {            return s2 == null ? 0 : 1;        } else if (s2 == null) {            return -1;        }        return s1.getName().compareTo(s2.getName());    });    Assertions.assertNotNull(students.get(0));    Assertions.assertNull(students.get(1));    Assertions.assertNull(students.get(2));}

    我们可以将判空的逻辑抽取出一个Comparator,通过组合方式实现:

    class NullComparator<T> implements Comparator<T> {    private final Comparator<T> real;    NullComparator(Comparator<? super T> real) {        this.real = (Comparator<T>) real;    }    @Override    public int compare(T a, T b) {        if (a == null) {            return (b == null) ? 0 : 1;        } else if (b == null) {            return -1;        } else {            return (real == null) ? 0 : real.compare(a, b);        }    }}

    在 Java8 中已经为我们准备了这个实现。

    使用Comparator.nullsLastComparator.nullsFirst

    使用Comparator.nullsLast实现null在结尾:

    @Testvoid sortedNullLast() {    final List<Student> students = Lists.newArrayList(            null,            new Student("Snoopy", 12),            null    );    students.sort(Comparator.nullsLast(Comparator.comparing(Student::getName)));    Assertions.assertNotNull(students.get(0));    Assertions.assertNull(students.get(1));    Assertions.assertNull(students.get(2));}

    使用Comparator.nullsFirst实现null在开头:

    @Testvoid sortedNullFirst() {    final List<Student> students = Lists.newArrayList(            null,            new Student("Snoopy", 12),            null    );    students.sort(Comparator.nullsFirst(Comparator.comparing(Student::getName)));    Assertions.assertNull(students.get(0));    Assertions.assertNull(students.get(1));    Assertions.assertNotNull(students.get(2));}

    是不是很简单,接下来我们看下如何实现排序条件的字段是 null 的逻辑。

    排序条件的字段是 null

    这个就是借助Comparator的组合了,就像是套娃实现了,需要使用两次Comparator.nullsLast,这里列出实现:

    @Testvoid sortedNullFieldLast() {    final List<Student> students = Lists.newArrayList(            new Student(null, 10),            new Student("Snoopy", 12),            null    );    final Comparator<Student> nullsLast = Comparator.nullsLast(            Comparator.nullsLast( // 1                    Comparator.comparing(                            Student::getName,                            Comparator.nullsLast( // 2                                    Comparator.naturalOrder() // 3                            )                    )            )    );    students.sort(nullsLast);    Assertions.assertEquals(students.get(0), new Student("Snoopy", 12));    Assertions.assertEquals(students.get(1), new Student(null, 10));    Assertions.assertNull(students.get(2));}

    代码逻辑如下:

    • 代码 1 是第一层 null-safe 逻辑,用于判断元素是否为 null;

    • 代码 2 是第二层 null-safe 逻辑,用于判断元素的条件字段是否为 null;

    • 代码 3 是条件Comparator,这里使用了Comparator.naturalOrder(),是因为使用了String排序,也可以写为String::compareTo。如果是复杂判断,可以定义一个更加复杂的Comparator,组合模式就是这么好用,一层不够再套一层。

    以上是“Java如何使用 Lambda 表达式实现超强的排序功能”这篇文章的所有内容,感谢各位的阅读!希望分享的内容对大家有帮助,更多相关知识,欢迎关注编程网精选频道!

    --结束END--

    本文标题: Java如何使用 Lambda 表达式实现超强的排序功能

    本文链接: https://www.lsjlt.com/news/305339.html(转载时请注明来源链接)

    有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

    本篇文章演示代码以及资料文档资料下载

    下载Word文档到电脑,方便收藏和打印~

    下载Word文档
    猜你喜欢
    • Java如何使用 Lambda 表达式实现超强的排序功能
      这篇文章主要介绍Java如何使用 Lambda 表达式实现超强的排序功能,文中介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们一定要看完!首先,我们定义一个基础类,后面我们将根据这个基础类演示如何在内存中排序。@Data@NoArgsC...
      99+
      2023-06-25
    • Java 进阶使用 Lambda 表达式实现超强的排序功能
      目录基于Comparator排序使用 Lambda 表达式替换Comparator匿名内部类通过静态方法抽取公共的 Lambda 表达式借助Comparator的comparing方...
      99+
      2022-11-12
    • Java中怎么用lambda表达式实现aop切面功能
      这篇“Java中怎么用lambda表达式实现aop切面功能”文章的知识点大部分人都不太理解,所以小编给大家总结了以下内容,内容详细,步骤清晰,具有一定的借鉴价值,希望大家阅读完这篇文章能有所收获,下面我们一起来看看这篇“Java中怎么用la...
      99+
      2023-06-29
    • Java的Lambda表达式如何使用
      这篇文章主要介绍“Java的Lambda表达式如何使用”,在日常操作中,相信很多人在Java的Lambda表达式如何使用问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”Java的Lambda表达式如何使用”的疑...
      99+
      2023-06-30
    • 使用Java 8中的Lambda表达式实现工厂模式
      前言工厂模式是面向对象设计模式中大家最为熟知的设计模式之一。传统的实现方式大家都在熟悉不过了,今天将向大家介绍使用Java8 Lambda 表达式更加优雅的实现工厂模式。封面工厂模式在java中最常用的设计模式之一,它提供了一种很好的实例化...
      99+
      2023-05-31
      java8 lambda表达式 工厂模式
    • 如何使用.NET Lambda表达式实现委托
      本篇内容介绍了“如何使用.NET Lambda表达式实现委托”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!.NET Lambda表达式的写法...
      99+
      2023-06-18
    • java使用lambda表达式对List对象集合的某个属性进行排序
      这里新建一个UserInfo对象,用来测试lambda表达式排序,属性如下:public class UserInfo { private int id; private int age; private Strin...
      99+
      2021-07-02
      java lambda List 对象集合 属性 排序
    • Java使用正则表达式如何实现查找文本功能
      本篇文章为大家展示了Java使用正则表达式如何实现查找文本功能,内容简明扼要并且容易理解,绝对能使你眼前一亮,通过这篇文章的详细介绍希望你能有所收获。具体如下:REMatch.java:package reMatch;import java...
      99+
      2023-05-31
      java 正则表达式 ava
    • Java如何使用正则表达式实现替换文本功能
      本篇文章给大家分享的是有关Java如何使用正则表达式实现替换文本功能,小编觉得挺实用的,因此分享给大家学习,希望大家阅读完这篇文章后可以有所收获,话不多说,跟着小编一起来看看吧。具体如下:package replaceDemo;import...
      99+
      2023-05-31
      java 正则表达式 ava
    • C#如何实现递归调用的Lambda表达式
      这篇文章主要讲解了“C#如何实现递归调用的Lambda表达式”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“C#如何实现递归调用的Lambda表达式”吧!首先给一个简单的示例: &n...
      99+
      2023-07-02
    • 利用PHP正则表达式函数实现强大的数据匹配功能
      利用PHP正则表达式函数实现强大的数据匹配功能正则表达式是一种强大的数据匹配工具,能够高效地对字符串进行模式匹配。在PHP中,正则表达式函数提供了许多功能,使得数据处理和筛选变得更加灵活和方便。一、正则表达式的基本语法正则表达式由一系列字符...
      99+
      2023-11-20
      正则表达式 关键词:PHP 数据匹配
    • 如何在Java中实现数组的排序功能?
      Java作为一门广泛应用于企业级开发的编程语言,提供了丰富的数据类型和数据结构,其中包括数组。而数组排序是Java中常见的操作之一,本篇文章将为你介绍如何在Java中实现数组排序功能。 Java中提供了两种排序方式:冒泡排序和快速排序。接下...
      99+
      2023-09-30
      数组 bash windows
    • 如何使用MongoDB实现数据排序功能
      如何使用MongoDB实现数据排序功能引言:MongoDB是一种非关系型数据库,它以文档的形式组织数据,并且提供了丰富的查询操作。在实际应用中,数据的排序是非常常见的需求之一。本文将介绍如何使用MongoDB实现数据排序功能,并提供具体的代...
      99+
      2023-10-22
      MongoDB 实现 数据排序
    • 微信小程序如何实现input正则表达式验证功能
      小编给大家分享一下微信小程序如何实现input正则表达式验证功能,相信大部分人都还不怎么了解,因此分享这篇文章给大家参考一下,希望大家阅读完这篇文章后大有收获,下面让我们一起去了解一下吧!具体如下:1、效果...
      99+
      2022-10-19
    • 如何使用MySQL和Java实现一个简单的排序算法功能
      如何使用MySQL和Java实现一个简单的排序算法功能导言:在软件开发中,排序算法是非常基础且常用的功能之一。本文将介绍如何使用MySQL和Java实现一个简单的排序算法功能,并提供具体代码示例。一、排序算法概述排序算法是将一组数据按照特定...
      99+
      2023-10-22
      排序算法 MySQL Java
    • react项目中使用react-dnd实现列表的拖拽排序功能
      目录1.先安装依赖2.创建一个 index.js 文件3.新建example.js文件4.新建TopicLis.js文件5.新建 ItemTypes.js现在有一个新需求就是需要对一...
      99+
      2023-02-06
      react-dnd列表的拖拽排序 react-dnd拖拽排序 react拖拽排序
    • 如何使用Redis的有序集合实现排行榜功能
      这篇文章给大家分享的是有关如何使用Redis的有序集合实现排行榜功能的内容。小编觉得挺实用的,因此分享给大家做个参考,一起跟随小编过来看看吧。一个典型的游戏排行榜包括以下常见功能:能够记录每个玩家的分数;能...
      99+
      2022-10-18
    • Java如何实现顺序表的增删查改功能
      这篇文章主要介绍Java如何实现顺序表的增删查改功能,文中介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们一定要看完!创建顺序表在java语言中要实现顺序表,首先创建一个类,因为顺序表本身就像数组,所以我们这里定义一个int类型的数组和...
      99+
      2023-06-14
    • Java使用正则表达式截取重复出现的XML字符串功能示例
      本文实例讲述了Java使用正则表达式截取重复出现的XML字符串功能。分享给大家供大家参考,具体如下:public static void main(String[] args) throws DocumentException { S...
      99+
      2023-05-31
      java 正则表达式 xml
    • python使用正则表达式的search()函数实现指定位置搜索功能
      前面学习过search()可以从任意一个文本里搜索匹配的字符串,也就是说可以从任何位置里搜索到匹配的字符串。但是现实世界很复杂多变的,比如限定你只能从第100个字符的位置开始匹配,100个字符之前的不要匹配...
      99+
      2022-06-04
      搜索功能 函数 位置
    软考高级职称资格查询
    编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
    • 官方手机版

    • 微信公众号

    • 商务合作