Java8 Comparator接口 和 List Steam 排序使用案例

在Java中,Comparator接口主要用于实现自定义排序逻辑,适用于未实现Comparable接口或需要覆盖默认比较规则的场景。以下是核心使用方法和注意事项:

一、基础用法

  1. 匿名内部类实现

    传统方式通过匿名内部类重写compare()方法,例如对整数降序排序:

    java 复制代码
    List<Integer> list = Arrays.asList(3, 1, 2); 
    list.sort(new Comparator<Integer>() { 
                @Override public int compare(Integer o1, Integer o2) {
                        // 降序 
                         return o2 - o1; 
                    } } );

    此方式在Java 8之前常见。

  2. Lambda表达式简化

    Java 8后可用Lambda替代匿名类,代码更简洁:

    java 复制代码
    list.sort((o1, o2) -> o2 - o1);

    或使用方法引用:

    java 复制代码
    list.sort(Comparator.reverseOrder());

    适用于简单排序场景。

二、高级特性

  1. 多条件排序

    使用thenComparing()实现多级排序,例如先按年龄升序,再按姓名降序:

    java 复制代码
    List<Person> people = ...; 
    people.sort( Comparator.comparing(Person::getAge) 
                          .thenComparing(Comparator.comparing(Person::getName)
                          .reversed() ) );

    支持链式调用,灵活组合排序规则。

  2. 处理null值

    需显式处理null避免NullPointerException,例如将null排在末尾:

    java 复制代码
    list.sort((a, b) -> {
                     if (a == null) return 1;
                     if (b == null) return -1; 
                    return a.compareTo(b);
                 });

    注意违反自反性会导致Comparison method violates its general contract错误。

三、Comparator与Comparable对比

维度 Comparator Comparable
实现位置 独立类或匿名类 目标类内部实现
排序规则 支持多种自定义规则 仅提供单一自然排序
侵入性 无侵入,不影响原有类 需修改类结构
典型应用场景 第三方类排序、多规则排序 类自身有明确排序逻辑(如String)

优先使用Comparator实现灵活排序,尤其是需要动态调整规则时。

四、常见问题

  1. 整数溢出风险
    避免直接返回o1 - o2,应使用

    java 复制代码
    Integer.compare(o1, o2)。
  2. JDK8增强方法

    • Comparator.naturalOrder():自然排序
    • Comparator.nullsFirst():将null置于开头
    • Comparator.comparingInt():避免装箱开销68。

通过合理应用Comparator,可高效实现复杂排序需求,同时保持代码可维护性。

Java 8的 Stream API 提供了强大的List排序功能,主要通过sorted()方法实现,支持自然排序、自定义排序以及多条件组合排序。以下是核心用法总结:

1. 基础排序

  • 自然排序(升序) ‌:直接调用sorted()方法,元素需实现Comparable接口

    java 复制代码
    list.stream().sorted().collect(Collectors.toList()); 
  • 降序排序 ‌:通过Comparator.reverseOrder()reversed()方法

    java 复制代码
    list.stream().sorted(Comparator.reverseOrder()).collect(Collectors.toList());
     // 或 
    list.stream().sorted(Comparator.comparing(Person::getAge).reversed())
                 .collect(Collectors.toList());

2. 自定义排序

  • 单字段排序 ‌:使用Comparator.comparing()指定字段

    java 复制代码
    list.stream().sorted(Comparator.comparing(Student::getScore))
            .collect(Collectors.toList());
  • 多字段排序 ‌:通过thenComparing()链式调用

    java 复制代码
    list.stream() .sorted(Comparator.comparing(Student::getAge)   
            .thenComparing(Student::getName)) .collect(Collectors.toList());

3. 空值处理

  • 空值优先/后置 ‌:使用Comparator.nullsFirst()Comparator.nullsLast()

    java 复制代码
    list.stream() .sorted(Comparator.comparing(Student::getAge,           
                               Comparator.nullsFirst(Integer::compareTo))) 
                .collect(Collectors.toList());

4. 性能提示

  • sorted()操作对大型流可能较昂贵,若数据已排序可跳过此操作2
  • 复杂排序建议预编译Comparator以减少重复计算9

完整示例:

java 复制代码
List<Student> students = Arrays.asList( 
        new Student("Alice", 25), 
        new Student("Bob", 20) );

List<Student> sorted = students.stream() 
                    .sorted(Comparator.comparing(Student::getAge).reversed()) 
                    .collect(Collectors.toList());
相关推荐
执笔论英雄8 小时前
【RL】async原理
java·服务器·前端
z***94848 小时前
Java进阶07 嵌套类
java·开发语言·python
python百炼成钢8 小时前
43.Linux LCD驱动
java·linux·运维·驱动开发
w***H6508 小时前
Springboot项目:使用MockMvc测试get和post接口(含单个和多个请求参数场景)
java·spring boot·后端
橘子编程8 小时前
仓颉语言:华为新一代编程利器
java·c语言·开发语言·数据库·python·青少年编程
a***13148 小时前
Spring Boot 条件注解:@ConditionalOnProperty 完全解析
java·spring boot·后端
axihaihai8 小时前
maven的构建问题
java·linux·maven
e***87708 小时前
Redis四种模式在Spring Boot框架下的配置
spring boot·redis·bootstrap
tgethe8 小时前
Java注解
java·后端
s***55819 小时前
IDEA+Docker一键部署项目SpringBoot项目
spring boot·docker·intellij-idea