Java Lambda
在系统开发过程中,对数据排序是很常见的场景。一般来说,可以采用两种方式:

  1. 借助存储系统(SQL、NoSQL、NewSQL 都支持)的排序功能,查询的结果即是排好序的结果
  2. 查询结果为无序数据,在内存中排序。

今天要说的是第二种排序方式,在内存中实现数据排序。
首先定义一个基础类,后面将根据这个基础类演示如何在内存中排序。

  1. @Data
  2. @NoArgsConstructor
  3. @AllArgsConstructor
  4. public class Student {
  5. private String name;
  6. private int age;
  7. @Override
  8. public boolean equals(Object o) {
  9. if (this == o) {
  10. return true;
  11. }
  12. if (o == null || getClass() != o.getClass()) {
  13. return false;
  14. }
  15. Student student = (Student) o;
  16. return age == student.age && Objects.equals(name, student.name);
  17. }
  18. @Override
  19. public int hashCode() {
  20. return Objects.hash(name, age);
  21. }
  22. }

基于Comparator排序

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

  1. new Comparator<Student>() {
  2. @Override
  3. public int compare(Student h1, Student h2) {
  4. return h1.getName().compareTo(h2.getName());
  5. }
  6. };

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

  1. @Test
  2. void baseSortedOrigin() {
  3. final List<Student> students = Lists.newArrayList(
  4. new Student("Tom", 10),
  5. new Student("Jerry", 12)
  6. );
  7. Collections.sort(students, new Comparator<Student>() {
  8. @Override
  9. public int compare(Student h1, Student h2) {
  10. return h1.getName().compareTo(h2.getName());
  11. }
  12. });
  13. Assertions.assertEquals(students.get(0), new Student("Jerry", 12));
  14. }

这里使用了 Junit5 实现单元测试,用来验证逻辑非常适合。
因为定义的Comparator是使用name字段排序,在 Java 中,String类型的排序是通过单字符的 ASCII 码顺序判断的,J排在T的前面,所以Jerry排在第一个。

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

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

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

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

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

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

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

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

  1. @Test
  2. void baseSortedLambdaWithInferring() {
  3. final List<Student> students = Lists.newArrayList(
  4. new Student("Tom", 10),
  5. new Student("Jerry", 12)
  6. );
  7. students.sort((h1, h2) -> h1.getName().compareTo(h2.getName()));
  8. Assertions.assertEquals(students.get(0), new Student("Jerry", 12));
  9. }

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

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

  1. public static int compareByNameThenAge(Student s1, Student s2) {
  2. if (s1.name.equals(s2.name)) {
  3. return Integer.compare(s1.age, s2.age);
  4. } else {
  5. return s1.name.compareTo(s2.name);
  6. }
  7. }

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

  1. @Test
  2. void sortedUsingStaticMethod() {
  3. final List<Student> students = Lists.newArrayList(
  4. new Student("Tom", 10),
  5. new Student("Jerry", 12)
  6. );
  7. students.sort(Student::compareByNameThenAge);
  8. Assertions.assertEquals(students.get(0), new Student("Jerry", 12));
  9. }

借助Comparatorcomparing方法

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

  1. @Test
  2. void sortedUsingComparator() {
  3. final List<Student> students = Lists.newArrayList(
  4. new Student("Tom", 10),
  5. new Student("Jerry", 12)
  6. );
  7. students.sort(Comparator.comparing(Student::getName));
  8. Assertions.assertEquals(students.get(0), new Student("Jerry", 12));
  9. }

多条件排序

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

  1. @Test
  2. void sortedMultiCondition() {
  3. final List<Student> students = Lists.newArrayList(
  4. new Student("Tom", 10),
  5. new Student("Jerry", 12),
  6. new Student("Jerry", 13)
  7. );
  8. students.sort((s1, s2) -> {
  9. if (s1.getName().equals(s2.getName())) {
  10. return Integer.compare(s1.getAge(), s2.getAge());
  11. } else {
  12. return s1.getName().compareTo(s2.getName());
  13. }
  14. });
  15. Assertions.assertEquals(students.get(0), new Student("Jerry", 12));
  16. }

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

  1. @Test
  2. void sortedMultiConditionUsingComparator() {
  3. final List<Student> students = Lists.newArrayList(
  4. new Student("Tom", 10),
  5. new Student("Jerry", 12),
  6. new Student("Jerry", 13)
  7. );
  8. students.sort(Comparator.comparing(Student::getName).thenComparing(Student::getAge));
  9. Assertions.assertEquals(students.get(0), new Student("Jerry", 12));
  10. }

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

在Stream中进行排序

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

  1. @Test
  2. void streamSorted() {
  3. final List<Student> students = Lists.newArrayList(
  4. new Student("Tom", 10),
  5. new Student("Jerry", 12)
  6. );
  7. final Comparator<Student> comparator = (h1, h2) -> h1.getName().compareTo(h2.getName());
  8. final List<Student> sortedStudents = students.stream()
  9. .sorted(comparator)
  10. .collect(Collectors.toList());
  11. Assertions.assertEquals(sortedStudents.get(0), new Student("Jerry", 12));
  12. }

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

  1. @Test
  2. void streamSortedUsingComparator() {
  3. final List<Student> students = Lists.newArrayList(
  4. new Student("Tom", 10),
  5. new Student("Jerry", 12)
  6. );
  7. final Comparator<Student> comparator = Comparator.comparing(Student::getName);
  8. final List<Student> sortedStudents = students.stream()
  9. .sorted(comparator)
  10. .collect(Collectors.toList());
  11. Assertions.assertEquals(sortedStudents.get(0), new Student("Jerry", 12));
  12. }

倒序排列

调转排序判断

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

  1. @Test
  2. void sortedReverseUsingComparator2() {
  3. final List<Student> students = Lists.newArrayList(
  4. new Student("Tom", 10),
  5. new Student("Jerry", 12)
  6. );
  7. final Comparator<Student> comparator = (h1, h2) -> h2.getName().compareTo(h1.getName());
  8. students.sort(comparator);
  9. Assertions.assertEquals(students.get(0), new Student("Tom", 10));
  10. }

可以看到,正序排列的时候,是h1.getName().compareTo(h2.getName()),这里直接倒转过来,使用的是h2.getName().compareTo(h1.getName()),也就达到了取反的效果。在 Java 的Collections中定义了一个java.util.Collections.ReverseComparator内部私有类,就是通过这种方式实现元素反转。
借助Comparatorreversed方法倒序
在 Java8 中新增了reversed方法实现倒序排列,用起来也是很简单:

  1. @Test
  2. void sortedReverseUsingComparator() {
  3. final List<Student> students = Lists.newArrayList(
  4. new Student("Tom", 10),
  5. new Student("Jerry", 12)
  6. );
  7. final Comparator<Student> comparator = (h1, h2) -> h1.getName().compareTo(h2.getName());
  8. students.sort(comparator.reversed());
  9. Assertions.assertEquals(students.get(0), new Student("Tom", 10));
  10. }

Comparator.comparing中定义排序反转

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

  1. @Test
  2. void sortedUsingComparatorReverse() {
  3. final List<Student> students = Lists.newArrayList(
  4. new Student("Tom", 10),
  5. new Student("Jerry", 12)
  6. );
  7. students.sort(Comparator.comparing(Student::getName, Comparator.reverseOrder()));
  8. Assertions.assertEquals(students.get(0), new Student("Jerry", 12));
  9. }

在Stream中定义排序反转

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

  1. @Test
  2. void streamReverseSorted() {
  3. final List<Student> students = Lists.newArrayList(
  4. new Student("Tom", 10),
  5. new Student("Jerry", 12)
  6. );
  7. final Comparator<Student> comparator = (h1, h2) -> h2.getName().compareTo(h1.getName());
  8. final List<Student> sortedStudents = students.stream()
  9. .sorted(comparator)
  10. .collect(Collectors.toList());
  11. Assertions.assertEquals(sortedStudents.get(0), new Student("Tom", 10));
  12. }
  13. @Test
  14. void streamReverseSortedUsingComparator() {
  15. final List<Student> students = Lists.newArrayList(
  16. new Student("Tom", 10),
  17. new Student("Jerry", 12)
  18. );
  19. final List<Student> sortedStudents = students.stream()
  20. .sorted(Comparator.comparing(Student::getName, Comparator.reverseOrder()))
  21. .collect(Collectors.toList());
  22. Assertions.assertEquals(sortedStudents.get(0), new Student("Tom", 10));
  23. }

null 值的判断

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

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

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

  1. @Test
  2. void sortedNullGotNPE() {
  3. final List<Student> students = Lists.newArrayList(
  4. null,
  5. new Student("Snoopy", 12),
  6. null
  7. );
  8. Assertions.assertThrows(NullPointerException.class,
  9. () -> students.sort(Comparator.comparing(Student::getName)));
  10. }

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

元素是 null 的笨拙实现

最先想到的就是判空:

  1. @Test
  2. void sortedNullNoNPE() {
  3. final List<Student> students = Lists.newArrayList(
  4. null,
  5. new Student("Snoopy", 12),
  6. null
  7. );
  8. students.sort((s1, s2) -> {
  9. if (s1 == null) {
  10. return s2 == null ? 0 : 1;
  11. } else if (s2 == null) {
  12. return -1;
  13. }
  14. return s1.getName().compareTo(s2.getName());
  15. });
  16. Assertions.assertNotNull(students.get(0));
  17. Assertions.assertNull(students.get(1));
  18. Assertions.assertNull(students.get(2));
  19. }

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

  1. class NullComparator<T> implements Comparator<T> {
  2. private final Comparator<T> real;
  3. NullComparator(Comparator<? super T> real) {
  4. this.real = (Comparator<T>) real;
  5. }
  6. @Override
  7. public int compare(T a, T b) {
  8. if (a == null) {
  9. return (b == null) ? 0 : 1;
  10. } else if (b == null) {
  11. return -1;
  12. } else {
  13. return (real == null) ? 0 : real.compare(a, b);
  14. }
  15. }
  16. }

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

使用Comparator.nullsLastComparator.nullsFirst

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

  1. @Test
  2. void sortedNullLast() {
  3. final List<Student> students = Lists.newArrayList(
  4. null,
  5. new Student("Snoopy", 12),
  6. null
  7. );
  8. students.sort(Comparator.nullsLast(Comparator.comparing(Student::getName)));
  9. Assertions.assertNotNull(students.get(0));
  10. Assertions.assertNull(students.get(1));
  11. Assertions.assertNull(students.get(2));
  12. }

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

  1. @Test
  2. void sortedNullFirst() {
  3. final List<Student> students = Lists.newArrayList(
  4. null,
  5. new Student("Snoopy", 12),
  6. null
  7. );
  8. students.sort(Comparator.nullsFirst(Comparator.comparing(Student::getName)));
  9. Assertions.assertNull(students.get(0));
  10. Assertions.assertNull(students.get(1));
  11. Assertions.assertNotNull(students.get(2));
  12. }

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

排序条件的字段是 null

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

  1. @Test
  2. void sortedNullFieldLast() {
  3. final List<Student> students = Lists.newArrayList(
  4. new Student(null, 10),
  5. new Student("Snoopy", 12),
  6. null
  7. );
  8. final Comparator<Student> nullsLast = Comparator.nullsLast(
  9. Comparator.nullsLast( // 1
  10. Comparator.comparing(
  11. Student::getName,
  12. Comparator.nullsLast( // 2
  13. Comparator.naturalOrder() // 3
  14. )
  15. )
  16. )
  17. );
  18. students.sort(nullsLast);
  19. Assertions.assertEquals(students.get(0), new Student("Snoopy", 12));
  20. Assertions.assertEquals(students.get(1), new Student(null, 10));
  21. Assertions.assertNull(students.get(2));
  22. }

代码逻辑如下:

  1. 代码 1 是第一层 null-safe 逻辑,用于判断元素是否为 null;
  2. 代码 2 是第二层 null-safe 逻辑,用于判断元素的条件字段是否为 null;
  3. 代码 3 是条件Comparator,这里使用了Comparator.naturalOrder(),是因为使用了String排序,也可以写为String::compareTo。如果是复杂判断,可以定义一个更加复杂的Comparator,组合模式就是这么好用,一层不够再套一层。